mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-25 03:29:41 +00:00 
			
		
		
		
	Compare commits
	
		
			432 Commits
		
	
	
		
			20211109.0
			...
			update-sta
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 2aeea10fc1 | ||
|   | fca7d2c5b0 | ||
|   | d7a5921e7b | ||
|   | cefa2ee183 | ||
|   | 0eeed85193 | ||
|   | fd80408de2 | ||
|   | 467a5169c0 | ||
|   | b0b3222b33 | ||
|   | b053881cef | ||
|   | 92a9ed7080 | ||
|   | 830b449006 | ||
|   | d38a8a317e | ||
|   | a0aed9112c | ||
|   | ce3b8544b9 | ||
|   | 134ed7d303 | ||
|   | dc27871189 | ||
|   | 9c9bfa2b77 | ||
|   | f02dd39619 | ||
|   | d37d99223d | ||
|   | 4db943c5ff | ||
|   | ed001fb10b | ||
|   | 5f43715dd8 | ||
|   | 5435218187 | ||
|   | 4ef5f3af89 | ||
|   | 9eea17b793 | ||
|   | 6a51e2aaad | ||
|   | 2d33327d88 | ||
|   | e9ec2da917 | ||
|   | 09d46dac61 | ||
|   | 236fa14ec3 | ||
|   | 2cb37820df | ||
|   | 869fa91ae5 | ||
|   | 22df03427f | ||
|   | e72a4e4a20 | ||
|   | ca8d31c6bb | ||
|   | 354ea88984 | ||
|   | 76af6e48cd | ||
|   | d05f807b9d | ||
|   | 4092f7f75d | ||
|   | 04668ad809 | ||
|   | 9be5a15c77 | ||
|   | 21d86f4797 | ||
|   | 45e6ec1ee2 | ||
|   | 9b97faa5e3 | ||
|   | 8730c122fd | ||
|   | 0046252e32 | ||
|   | f47440083e | ||
|   | bfaf44f9d1 | ||
|   | deba6a0db4 | ||
|   | 890ad9a1c8 | ||
|   | 8466ef371a | ||
|   | 4e55460799 | ||
|   | 5fde6e659d | ||
|   | 148bb99d89 | ||
|   | 0540bae707 | ||
|   | 0c6f647f53 | ||
|   | 3aca67d511 | ||
|   | 0e41a408e7 | ||
|   | 19e1eaf2d7 | ||
|   | 5e80a2b465 | ||
|   | 866a57cde4 | ||
|   | a88da0e39a | ||
|   | 21a8fac477 | ||
|   | ca5ce04a38 | ||
|   | 7c4b9a0410 | ||
|   | de6f06ea6d | ||
|   | bbc8e323e8 | ||
|   | 89b6863ae3 | ||
|   | 3f1850e9eb | ||
|   | 54d6b5b6f3 | ||
|   | fb55ab197f | ||
|   | cc2db9a761 | ||
|   | 58ba3e5c22 | ||
|   | 182ffccd0c | ||
|   | ce99d14ee0 | ||
|   | 8ce160b9ce | ||
|   | fe33714c8b | ||
|   | afbe85625c | ||
|   | cb47ee7721 | ||
|   | 5caa256f1b | ||
|   | c66dfb84f9 | ||
|   | df1d703e4e | ||
|   | ce0ced0b6a | ||
|   | 730e9b144d | ||
|   | 69ff8dd0c4 | ||
|   | 8d2c716fbe | ||
|   | 389a100b46 | ||
|   | 9fee7a2829 | ||
|   | a91897821a | ||
|   | 815a2a07ff | ||
|   | b8d3eb76ac | ||
|   | ba75c2e7af | ||
|   | f04b844223 | ||
|   | 242bad0a29 | ||
|   | 8b20b2b63c | ||
|   | e0c8efc5e6 | ||
|   | f59c30ac04 | ||
|   | e4b9c08b45 | ||
|   | 04e63eefe2 | ||
|   | a064ca0856 | ||
|   | 6044ea92ad | ||
|   | 17e8215420 | ||
|   | a4ae1bee79 | ||
|   | 7d335d7d85 | ||
|   | 7c194d8910 | ||
|   | a92100bb0a | ||
|   | 303af611d1 | ||
|   | 559b6e9d5b | ||
|   | 75a95ff675 | ||
|   | 3024ee43f9 | ||
|   | b34b92fa87 | ||
|   | 1832ed0a48 | ||
|   | f398692e75 | ||
|   | 68bee4dd58 | ||
|   | f1297e1f36 | ||
|   | 953e3e060b | ||
|   | c37f660718 | ||
|   | 02754369a6 | ||
|   | 0df9e9932f | ||
|   | eddb392ad0 | ||
|   | e8ba349447 | ||
|   | 5be22d46ab | ||
|   | ffaff30b46 | ||
|   | c4cad5bccd | ||
|   | e4085fe1f6 | ||
|   | 8bfef92c86 | ||
|   | 0c07178c0a | ||
|   | 1010777139 | ||
|   | e57477c16a | ||
|   | 30fa92c120 | ||
|   | b32438dc18 | ||
|   | 614bd2f451 | ||
|   | 6c12a5a4b1 | ||
|   | bbcec38450 | ||
|   | 416e2e26c0 | ||
|   | 1a7164b466 | ||
|   | 3ddcd2d0f6 | ||
|   | 648c02e622 | ||
|   | b0b953bfac | ||
|   | abeaa63005 | ||
|   | 9cd23374f4 | ||
|   | 72bd5f84d6 | ||
|   | 22b4550fdf | ||
|   | 87c22229e0 | ||
|   | 971fd8dc60 | ||
|   | 049c3caadd | ||
|   | fb2a24d11e | ||
|   | d4646bac01 | ||
|   | 14e5b2a7a5 | ||
|   | 734a733a4c | ||
|   | 8f31c182f6 | ||
|   | e51a819bfd | ||
|   | 05d7e85aa3 | ||
|   | cf527e4bc2 | ||
|   | 197b581e8e | ||
|   | f75bf1f676 | ||
|   | 28df79cfda | ||
|   | 3bf19883a8 | ||
|   | 303e065433 | ||
|   | 7ad0b37a9e | ||
|   | 930c7e4afa | ||
|   | 81faae6f74 | ||
|   | f7fc83ac12 | ||
|   | 21a099ee9f | ||
|   | 7d1ce1b240 | ||
|   | d1f1309198 | ||
|   | 68dd818f7a | ||
|   | 50bea33a19 | ||
|   | 27cae037ce | ||
|   | dbb5bf7550 | ||
|   | 9ef743a695 | ||
|   | f3642a1677 | ||
|   | 2d651c2a66 | ||
|   | ef39317019 | ||
|   | 441f1fbcb5 | ||
|   | 09a27a6791 | ||
|   | 32bbdc194a | ||
|   | 52588a3915 | ||
|   | 9fffc93e5d | ||
|   | effec839af | ||
|   | 884ed561a1 | ||
|   | 4165e64ce0 | ||
|   | 6053b64b2e | ||
|   | ed462dc257 | ||
|   | 74a05929be | ||
|   | 5e388b1f02 | ||
|   | ff2fa9a78c | ||
|   | 6d1be9e73f | ||
|   | ba570f4004 | ||
|   | 6ab497edf8 | ||
|   | 8c1cd273df | ||
|   | 8f68bcbba9 | ||
|   | 8291cf9daa | ||
|   | bb40e66833 | ||
|   | f852208eff | ||
|   | 3bbe1603eb | ||
|   | 25d60e11da | ||
|   | 78d06426cf | ||
|   | 320b2bb48b | ||
|   | a7b8382617 | ||
|   | 77fe687ec2 | ||
|   | 069f08b55e | ||
|   | 204ccf8b40 | ||
|   | 4b9ff641ba | ||
|   | 1520b5832a | ||
|   | 04f2e2e70c | ||
|   | 920d2972ea | ||
|   | e94fc493b8 | ||
|   | 3e22270c2c | ||
|   | 27fa34e24e | ||
|   | 4e0cebaf32 | ||
|   | f021480bc5 | ||
|   | 34c3374d84 | ||
|   | 4cb7154917 | ||
|   | 08863348dc | ||
|   | 2bcf816b77 | ||
|   | d2b99e6963 | ||
|   | 48a800882e | ||
|   | 595e13ecac | ||
|   | 5261d583a8 | ||
|   | 5c488f8298 | ||
|   | 6fc87a6f66 | ||
|   | 3133f9b01f | ||
|   | 2c0d330f1f | ||
|   | fb9ea981ed | ||
|   | 63c113f78d | ||
|   | a67799a670 | ||
|   | e3d78d6dc5 | ||
|   | 76a4b1efbd | ||
|   | 882e79524b | ||
|   | 0ab8f8fd7c | ||
|   | 86b9eb0bd7 | ||
|   | 011cbe7d22 | ||
|   | 9b0b2c5b71 | ||
|   | be72bf7b3c | ||
|   | 3e062ba673 | ||
|   | 322d965539 | ||
|   | 7b840527b5 | ||
|   | dced053ba2 | ||
|   | fe4322e64b | ||
|   | 0800c702fb | ||
|   | b6d6e2fd4b | ||
|   | 2bbb1bfa7e | ||
|   | e2af8ac3cc | ||
|   | 25ff5fef14 | ||
|   | 2f9c088091 | ||
|   | 50c397901b | ||
|   | 1f7d4c25d4 | ||
|   | 29819fac23 | ||
|   | cc301df57d | ||
|   | 7d5b566312 | ||
|   | 07cd68f5d0 | ||
|   | 99bf6fa781 | ||
|   | bfad1eb5ac | ||
|   | 6f9b2ee569 | ||
|   | b7bd7c1065 | ||
|   | 4ebdca2a46 | ||
|   | fc700fdaf0 | ||
|   | d8e12f4280 | ||
|   | 86114758c3 | ||
|   | 792278cf17 | ||
|   | b8832f2121 | ||
|   | 76339c90f7 | ||
|   | b3d4451035 | ||
|   | dc58481918 | ||
|   | 14af735507 | ||
|   | a7b558b64a | ||
|   | b7665bef6f | ||
|   | 5ec37a35f1 | ||
|   | 91bb2ddcc4 | ||
|   | 61bae5da64 | ||
|   | 85168b3a35 | ||
|   | 942150cda2 | ||
|   | 2606d55895 | ||
|   | 1f671198aa | ||
|   | deb65e7108 | ||
|   | cd00f7f874 | ||
|   | bdd13db8cf | ||
|   | 2b0359edba | ||
|   | 35e9687170 | ||
|   | b730676914 | ||
|   | 2890192c05 | ||
|   | bfb84a834f | ||
|   | ca6fd6c770 | ||
|   | 585648ac4c | ||
|   | bec5c564b6 | ||
|   | 48c66e6349 | ||
|   | cea40610c0 | ||
|   | 0c3fd8f3ad | ||
|   | cdc3d11181 | ||
|   | 02bdeebc82 | ||
|   | 60c7669d8f | ||
|   | 919bf94a03 | ||
|   | ead5e288eb | ||
|   | add8a702cc | ||
|   | 39774c0e02 | ||
|   | 149f381bc3 | ||
|   | faccb12430 | ||
|   | 7039bae9be | ||
|   | 0a7b703d57 | ||
|   | 24e8028e8f | ||
|   | 8f729e2a95 | ||
|   | 8412cd71cb | ||
|   | 5c78b74005 | ||
|   | 2459477ec4 | ||
|   | a065740c91 | ||
|   | f3104d3c93 | ||
|   | 1916c179b4 | ||
|   | e8b9766eb6 | ||
|   | ff7a2c8cb7 | ||
|   | 7ccde2cb41 | ||
|   | d6b9b16f02 | ||
|   | 66df15007a | ||
|   | f164d21c44 | ||
|   | 911d322aac | ||
|   | 419879ee7a | ||
|   | c3e1a2edf0 | ||
|   | 8f5751d5bb | ||
|   | 4095450476 | ||
|   | bc9195f7d5 | ||
|   | e61f587c51 | ||
|   | d43d19190e | ||
|   | a283acaabf | ||
|   | ea18fc0078 | ||
|   | 1df11e9bf1 | ||
|   | c71b2e6b9d | ||
|   | db4aa05bf4 | ||
|   | a54a2a54f8 | ||
|   | 0bcb4d0e09 | ||
|   | 95dbc811d3 | ||
|   | e28a11964e | ||
|   | 46a9e36516 | ||
|   | e99f20c4f3 | ||
|   | 2100603cdc | ||
|   | da4942aca3 | ||
|   | 7c78fb314e | ||
|   | 5bc2468cbc | ||
|   | a580904c52 | ||
|   | 48d12ceafe | ||
|   | 60ce805b3b | ||
|   | 251416b51d | ||
|   | c41c6eedd8 | ||
|   | 6877fd9e00 | ||
|   | 4cc104a99f | ||
|   | 6494177821 | ||
|   | cea1a62867 | ||
|   | a6b5262d02 | ||
|   | 2a5fc5181e | ||
|   | 2fe8f5ff27 | ||
|   | 0c75d5afc9 | ||
|   | cf062bf0f4 | ||
|   | acf4d59fde | ||
|   | 05333ac2d9 | ||
|   | 4b49da58b1 | ||
|   | 68373e6372 | ||
|   | 01049e8eb8 | ||
|   | 87f7981144 | ||
|   | ceac9834b9 | ||
|   | ac8f748656 | ||
|   | 1d97d8dca9 | ||
|   | fd6785b593 | ||
|   | d5fc751da6 | ||
|   | 933fd72629 | ||
|   | 0611133065 | ||
|   | 02644b923f | ||
|   | 67f06112c6 | ||
|   | 49e39644f3 | ||
|   | 990ad1bb67 | ||
|   | dbbf246060 | ||
|   | d2c20837a5 | ||
|   | e91d1777d0 | ||
|   | a5be143c3b | ||
|   | 0ef07e4835 | ||
|   | 9361e4cf9c | ||
|   | e7fd75703f | ||
|   | 2c0b2f4bc5 | ||
|   | faec09f0d1 | ||
|   | b79c06ad71 | ||
|   | 5614e0d29c | ||
|   | 0b7fc177f9 | ||
|   | 367322415e | ||
|   | 117b50f3ea | ||
|   | 366aa8aed1 | ||
|   | 43011179eb | ||
|   | 6177d2b416 | ||
|   | f70485bc49 | ||
|   | 921763b5f1 | ||
|   | 5fd4315789 | ||
|   | ed291b57d0 | ||
|   | f833701e7c | ||
|   | 8533b90957 | ||
|   | c95a54c6f3 | ||
|   | a991640f52 | ||
|   | 3d99b92c07 | ||
|   | d28ad17135 | ||
|   | 3c67fc96b1 | ||
|   | 4719636176 | ||
|   | 45efee28b8 | ||
|   | 3bcf225380 | ||
|   | 2e81f843ce | ||
|   | a430142296 | ||
|   | 6335b13c5e | ||
|   | 6c4e987a24 | ||
|   | 1a5c43d72a | ||
|   | 91dbfca899 | ||
|   | 96f103644a | ||
|   | 5304e5a670 | ||
|   | 390e5b3881 | ||
|   | 9f5756c9fa | ||
|   | 0ca35d7012 | ||
|   | 0d19f4792f | ||
|   | 91b009af79 | ||
|   | 1ebd2fb9f1 | ||
|   | 4684979ae7 | ||
|   | a567312bdb | ||
|   | 1e851e0e8c | ||
|   | 7d94615f47 | ||
|   | 582fab7ea1 | ||
|   | 822590ec8a | ||
|   | e9f0967578 | ||
|   | 481da19c74 | ||
|   | b969db0c0f | ||
|   | a6b98fc3c3 | ||
|   | 87c2046ab5 | ||
|   | 4b992fb0c4 | ||
|   | 3154011c65 | ||
|   | 4e68383cf7 | ||
|   | db6ef22ebb | ||
|   | c238c7dbbc | ||
|   | d04823b4c5 | ||
|   | 4cb45d6313 | ||
|   | 6623e5f017 | 
							
								
								
									
										2
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -30,7 +30,7 @@ jobs: | ||||
|         env: | ||||
|           CI: true | ||||
|       - name: Build resources | ||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-demos | ||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages | ||||
|       - name: Run eslint | ||||
|         run: yarn run lint:eslint | ||||
|       - name: Run tsc | ||||
|   | ||||
							
								
								
									
										4
									
								
								.github/workflows/netflify.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.github/workflows/netflify.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,5 +15,5 @@ jobs: | ||||
|       - name: Trigger Demo build | ||||
|         run: curl -X POST -d {} https://api.netlify.com/build_hooks/${{ secrets.NETLIFY_DEMO_DEV_BUILD_HOOK }} | ||||
|  | ||||
|       - name: Trigger Gallery build | ||||
|         run: curl -X POST -d {} https://api.netlify.com/build_hooks/${{ secrets.NETLIFY_GALLERY_DEV_BUILD_HOOK }} | ||||
|       - name: Trigger Design build | ||||
|         run: curl -X POST -d "NIGHTLY" https://api.netlify.com/build_hooks/${{ secrets.NETLIFY_GALLERY_DEV_BUILD_HOOK }} | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -41,7 +41,7 @@ jobs: | ||||
|           LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }} | ||||
|       - name: Build and release package | ||||
|         run: | | ||||
|           python3 -m pip install twine | ||||
|           python3 -m pip install twine build | ||||
|           export TWINE_USERNAME="__token__" | ||||
|           export TWINE_PASSWORD="${{ secrets.TWINE_TOKEN }}" | ||||
|  | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -1,11 +1,10 @@ | ||||
| diff --git a/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js b/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js | ||||
| index d92179f7fd5315203f870a6963e871dc8ddf6c0c..362e284121b97e0fba0925225777aebc32e26b8d 100644 | ||||
| --- a/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js | ||||
| +++ b/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js | ||||
| @@ -1,14 +1,15 @@ | ||||
| -let _ET, ET; | ||||
| +let _ET; | ||||
| +let ET; | ||||
| diff --git a/polyfillLoaders/EventTarget.js b/polyfillLoaders/EventTarget.js | ||||
| index 4e18ade7ba485849f17f28c94c42f0e0e01ac387..8f34f4f646c7f7becc208fb5a546c96034fc74dc 100644 | ||||
| --- a/polyfillLoaders/EventTarget.js | ||||
| +++ b/polyfillLoaders/EventTarget.js | ||||
| @@ -6,16 +6,15 @@ | ||||
|  let _ET; | ||||
|  let ET; | ||||
|  export default async function EventTarget() { | ||||
| -    return ET || init(); | ||||
| +  return ET || init(); | ||||
| @@ -27,3 +26,4 @@ index d92179f7fd5315203f870a6963e871dc8ddf6c0c..362e284121b97e0fba0925225777aebc | ||||
| +  } | ||||
| +  return (ET = _ET); | ||||
|  } | ||||
|  //# sourceMappingURL=EventTarget.js.map | ||||
| @@ -1,5 +1,4 @@ | ||||
| include README.md | ||||
| include LICENSE.md | ||||
| graft hass_frontend | ||||
| graft hass_frontend_es5 | ||||
| recursive-exclude * *.py[co] | ||||
|   | ||||
| @@ -10,7 +10,7 @@ module.exports.ignorePackages = ({ latestBuild }) => [ | ||||
| ]; | ||||
|  | ||||
| // Files from NPM packages that we should replace with empty file | ||||
| module.exports.emptyPackages = ({ latestBuild }) => | ||||
| module.exports.emptyPackages = ({ latestBuild, isHassioBuild }) => | ||||
|   [ | ||||
|     // Contains all color definitions for all material color sets. | ||||
|     // We don't use it | ||||
| @@ -28,6 +28,11 @@ module.exports.emptyPackages = ({ latestBuild }) => | ||||
|       ), | ||||
|     // This polyfill is loaded in workers to support ES5, filter it out. | ||||
|     latestBuild && require.resolve("proxy-polyfill/src/index.js"), | ||||
|     // Icons in supervisor conflict with icons in HA so we don't load. | ||||
|     isHassioBuild && | ||||
|       require.resolve( | ||||
|         path.resolve(paths.polymer_dir, "src/components/ha-icon.ts") | ||||
|       ), | ||||
|   ].filter(Boolean); | ||||
|  | ||||
| module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ | ||||
| @@ -196,6 +201,7 @@ module.exports.config = { | ||||
|       publicPath: publicPath(latestBuild, paths.hassio_publicPath), | ||||
|       isProdBuild, | ||||
|       latestBuild, | ||||
|       isHassioBuild: true, | ||||
|       defineOverlay: { | ||||
|         __SUPERVISOR__: true, | ||||
|       }, | ||||
|   | ||||
| @@ -26,11 +26,11 @@ module.exports = { | ||||
|   }, | ||||
|   version() { | ||||
|     const version = fs | ||||
|       .readFileSync(path.resolve(paths.polymer_dir, "setup.py"), "utf8") | ||||
|       .match(/\d{8}\.\d+/); | ||||
|       .readFileSync(path.resolve(paths.polymer_dir, "setup.cfg"), "utf8") | ||||
|       .match(/version\W+=\W(\d{8}\.\d)/); | ||||
|     if (!version) { | ||||
|       throw Error("Version not found"); | ||||
|     } | ||||
|     return version[0]; | ||||
|     return version[1]; | ||||
|   }, | ||||
| }; | ||||
|   | ||||
| @@ -31,6 +31,6 @@ gulp.task("clean-hassio", () => | ||||
| gulp.task( | ||||
|   "clean-gallery", | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.gallery_output_root, paths.build_dir]) | ||||
|     del([paths.gallery_output_root, paths.gallery_build, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|   | ||||
| @@ -1,7 +1,11 @@ | ||||
| /* eslint-disable */ | ||||
| // Run demo develop mode | ||||
| const gulp = require("gulp"); | ||||
| const fs = require("fs"); | ||||
| const path = require("path"); | ||||
| const marked = require("marked"); | ||||
| const glob = require("glob"); | ||||
| const yaml = require("js-yaml"); | ||||
|  | ||||
| const env = require("../env"); | ||||
| const paths = require("../paths"); | ||||
| @@ -15,26 +19,129 @@ require("./service-worker.js"); | ||||
| require("./entry-html.js"); | ||||
| require("./rollup.js"); | ||||
|  | ||||
| gulp.task("gather-gallery-demos", async function gatherDemos() { | ||||
|   const files = await fs.promises.readdir( | ||||
|     path.resolve(paths.gallery_dir, "src/demos") | ||||
|   ); | ||||
|  | ||||
|   let content = "export const DEMOS = {\n"; | ||||
|  | ||||
|   for (const file of files) { | ||||
|     const demoId = path.basename(file, ".ts"); | ||||
|     const demoPath = "../src/demos/" + demoId; | ||||
|     content += `  "${demoId}": () => import("${demoPath}"),\n`; | ||||
|   } | ||||
|  | ||||
|   content += "};"; | ||||
| gulp.task("gather-gallery-pages", async function gatherPages() { | ||||
|   const pageDir = path.resolve(paths.gallery_dir, "src/pages"); | ||||
|   const files = glob.sync(path.resolve(pageDir, "**/*")); | ||||
|  | ||||
|   const galleryBuild = path.resolve(paths.gallery_dir, "build"); | ||||
|  | ||||
|   fs.mkdirSync(galleryBuild, { recursive: true }); | ||||
|  | ||||
|   let content = "export const PAGES = {\n"; | ||||
|  | ||||
|   const processed = new Set(); | ||||
|  | ||||
|   for (const file of files) { | ||||
|     if (fs.lstatSync(file).isDirectory()) { | ||||
|       continue; | ||||
|     } | ||||
|     const pageId = file.substring(pageDir.length + 1, file.lastIndexOf(".")); | ||||
|  | ||||
|     if (processed.has(pageId)) { | ||||
|       continue; | ||||
|     } | ||||
|     processed.add(pageId); | ||||
|  | ||||
|     const [category, name] = pageId.split("/", 2); | ||||
|  | ||||
|     const demoFile = path.resolve(pageDir, `${pageId}.ts`); | ||||
|     const descriptionFile = path.resolve(pageDir, `${pageId}.markdown`); | ||||
|     const hasDemo = fs.existsSync(demoFile); | ||||
|     let hasDescription = fs.existsSync(descriptionFile); | ||||
|     let metadata = {}; | ||||
|     if (hasDescription) { | ||||
|       let descriptionContent = fs.readFileSync(descriptionFile, "utf-8"); | ||||
|  | ||||
|       if (descriptionContent.startsWith("---")) { | ||||
|         const metadataEnd = descriptionContent.indexOf("---", 3); | ||||
|         metadata = yaml.load(descriptionContent.substring(3, metadataEnd)); | ||||
|         descriptionContent = descriptionContent | ||||
|           .substring(metadataEnd + 3) | ||||
|           .trim(); | ||||
|       } | ||||
|  | ||||
|       // If description is just metadata | ||||
|       if (descriptionContent === "") { | ||||
|         hasDescription = false; | ||||
|       } else { | ||||
|         descriptionContent = marked(descriptionContent).replace(/`/g, "\\`"); | ||||
|         fs.mkdirSync(path.resolve(galleryBuild, category), { recursive: true }); | ||||
|         fs.writeFileSync( | ||||
|     path.resolve(galleryBuild, "import-demos.ts"), | ||||
|           path.resolve(galleryBuild, `${pageId}-description.ts`), | ||||
|           ` | ||||
|           import {html} from "lit"; | ||||
|           export default html\`${descriptionContent}\` | ||||
|           ` | ||||
|         ); | ||||
|       } | ||||
|     } | ||||
|     content += `  "${pageId}": { | ||||
|       metadata: ${JSON.stringify(metadata)}, | ||||
|       ${ | ||||
|         hasDescription | ||||
|           ? `description: () => import("./${pageId}-description").then(m => m.default),` | ||||
|           : "" | ||||
|       } | ||||
|       ${hasDemo ? `demo: () => import("../src/pages/${pageId}")` : ""} | ||||
|  | ||||
|     },\n`; | ||||
|   } | ||||
|  | ||||
|   content += "};\n"; | ||||
|  | ||||
|   // Generate sidebar | ||||
|   const sidebarPath = path.resolve(paths.gallery_dir, "sidebar.js"); | ||||
|   // To make watch work during development | ||||
|   delete require.cache[sidebarPath]; | ||||
|   const sidebar = require(sidebarPath); | ||||
|  | ||||
|   const pagesToProcess = {}; | ||||
|   for (const key of processed) { | ||||
|     const [category, page] = key.split("/", 2); | ||||
|     if (!(category in pagesToProcess)) { | ||||
|       pagesToProcess[category] = new Set(); | ||||
|     } | ||||
|     pagesToProcess[category].add(page); | ||||
|   } | ||||
|  | ||||
|   for (const group of Object.values(sidebar)) { | ||||
|     const toProcess = pagesToProcess[group.category]; | ||||
|     delete pagesToProcess[group.category]; | ||||
|  | ||||
|     if (!toProcess) { | ||||
|       console.error("Unknown category", group.category); | ||||
|       if (!group.pages) { | ||||
|         group.pages = []; | ||||
|       } | ||||
|       continue; | ||||
|     } | ||||
|  | ||||
|     // Any pre-defined groups will not be sorted. | ||||
|     if (group.pages) { | ||||
|       for (const page of group.pages) { | ||||
|         if (!toProcess.delete(page)) { | ||||
|           console.error("Found unreferenced demo", page); | ||||
|         } | ||||
|       } | ||||
|     } else { | ||||
|       group.pages = []; | ||||
|     } | ||||
|     for (const page of Array.from(toProcess).sort()) { | ||||
|       group.pages.push(page); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   for (const [category, pages] of Object.entries(pagesToProcess)) { | ||||
|     sidebar.push({ | ||||
|       category, | ||||
|       header: category, | ||||
|       pages: Array.from(pages).sort(), | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   content += `export const SIDEBAR = ${JSON.stringify(sidebar, null, 2)};\n`; | ||||
|  | ||||
|   fs.writeFileSync( | ||||
|     path.resolve(galleryBuild, "import-pages.ts"), | ||||
|     content, | ||||
|     "utf-8" | ||||
|   ); | ||||
| @@ -52,11 +159,24 @@ gulp.task( | ||||
|       "gen-icons-json", | ||||
|       "build-translations", | ||||
|       "build-locale-data", | ||||
|       "gather-gallery-demos" | ||||
|       "gather-gallery-pages" | ||||
|     ), | ||||
|     "copy-static-gallery", | ||||
|     "gen-index-gallery-dev", | ||||
|     env.useRollup() ? "rollup-dev-server-gallery" : "webpack-dev-server-gallery" | ||||
|     gulp.parallel( | ||||
|       env.useRollup() | ||||
|         ? "rollup-dev-server-gallery" | ||||
|         : "webpack-dev-server-gallery", | ||||
|       async function watchMarkdownFiles() { | ||||
|         gulp.watch( | ||||
|           [ | ||||
|             path.resolve(paths.gallery_dir, "src/pages/**/*.markdown"), | ||||
|             path.resolve(paths.gallery_dir, "sidebar.js"), | ||||
|           ], | ||||
|           gulp.series("gather-gallery-pages") | ||||
|         ); | ||||
|       } | ||||
|     ) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| @@ -72,7 +192,7 @@ gulp.task( | ||||
|       "gen-icons-json", | ||||
|       "build-translations", | ||||
|       "build-locale-data", | ||||
|       "gather-gallery-demos" | ||||
|       "gather-gallery-pages" | ||||
|     ), | ||||
|     "copy-static-gallery", | ||||
|     env.useRollup() ? "rollup-prod-gallery" : "webpack-prod-gallery", | ||||
|   | ||||
| @@ -79,6 +79,11 @@ function copyFonts(staticDir) { | ||||
|   ); | ||||
| } | ||||
|  | ||||
| function copyQrScannerWorker(staticDir) { | ||||
|   const staticPath = genStaticPath(staticDir); | ||||
|   copyFileDir(npmPath("qr-scanner/qr-scanner-worker.min.js"), staticPath("js")); | ||||
| } | ||||
|  | ||||
| function copyMapPanel(staticDir) { | ||||
|   const staticPath = genStaticPath(staticDir); | ||||
|   copyFileDir( | ||||
| @@ -125,6 +130,9 @@ gulp.task("copy-static-app", async () => { | ||||
|  | ||||
|   // Panel assets | ||||
|   copyMapPanel(staticDir); | ||||
|  | ||||
|   // Qr Scanner assets | ||||
|   copyQrScannerWorker(staticDir); | ||||
| }); | ||||
|  | ||||
| gulp.task("copy-static-demo", async () => { | ||||
|   | ||||
| @@ -26,6 +26,7 @@ module.exports = { | ||||
|   cast_output_es5: path.resolve(__dirname, "../cast/dist/frontend_es5"), | ||||
|  | ||||
|   gallery_dir: path.resolve(__dirname, "../gallery"), | ||||
|   gallery_build: path.resolve(__dirname, "../gallery/build"), | ||||
|   gallery_output_root: path.resolve(__dirname, "../gallery/dist"), | ||||
|   gallery_output_latest: path.resolve( | ||||
|     __dirname, | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -30,6 +30,7 @@ const createWebpackConfig = ({ | ||||
|   isProdBuild, | ||||
|   latestBuild, | ||||
|   isStatsBuild, | ||||
|   isHassioBuild, | ||||
|   dontHash, | ||||
| }) => { | ||||
|   if (!dontHash) { | ||||
| @@ -117,7 +118,9 @@ const createWebpackConfig = ({ | ||||
|         }, | ||||
|       }), | ||||
|       new webpack.NormalModuleReplacementPlugin( | ||||
|         new RegExp(bundle.emptyPackages({ latestBuild }).join("|")), | ||||
|         new RegExp( | ||||
|           bundle.emptyPackages({ latestBuild, isHassioBuild }).join("|") | ||||
|         ), | ||||
|         path.resolve(paths.polymer_dir, "src/util/empty.js") | ||||
|       ), | ||||
|       !isProdBuild && new LogStartCompilePlugin(), | ||||
|   | ||||
| @@ -7,6 +7,9 @@ import "../../../../src/panels/lovelace/views/hui-view"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import "./hc-launch-screen"; | ||||
|  | ||||
| (window as any).loadCardHelpers = () => | ||||
|   import("../../../../src/panels/lovelace/custom-card-helpers"); | ||||
|  | ||||
| @customElement("hc-lovelace") | ||||
| class HcLovelace extends LitElement { | ||||
|   @property({ attribute: false }) public hass!: HomeAssistant; | ||||
| @@ -15,7 +18,7 @@ class HcLovelace extends LitElement { | ||||
|  | ||||
|   @property() public viewPath?: string | number; | ||||
|  | ||||
|   public urlPath?: string | null; | ||||
|   @property() public urlPath: string | null = null; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     const index = this._viewIndex; | ||||
| @@ -31,7 +34,7 @@ class HcLovelace extends LitElement { | ||||
|       config: this.lovelaceConfig, | ||||
|       rawConfig: this.lovelaceConfig, | ||||
|       editMode: false, | ||||
|       urlPath: this.urlPath!, | ||||
|       urlPath: this.urlPath, | ||||
|       enableFullEditMode: () => undefined, | ||||
|       mode: "storage", | ||||
|       locale: this.hass.locale, | ||||
|   | ||||
| @@ -13,7 +13,11 @@ import { | ||||
|   ShowDemoMessage, | ||||
|   ShowLovelaceViewMessage, | ||||
| } from "../../../../src/cast/receiver_messages"; | ||||
| import { ReceiverStatusMessage } from "../../../../src/cast/sender_messages"; | ||||
| import { | ||||
|   ReceiverErrorCode, | ||||
|   ReceiverErrorMessage, | ||||
|   ReceiverStatusMessage, | ||||
| } from "../../../../src/cast/sender_messages"; | ||||
| import { atLeastVersion } from "../../../../src/common/config/version"; | ||||
| import { isNavigationClick } from "../../../../src/common/dom/is-navigation-click"; | ||||
| import { | ||||
| @@ -40,9 +44,9 @@ export class HcMain extends HassElement { | ||||
|  | ||||
|   @state() private _error?: string; | ||||
|  | ||||
|   private _unsubLovelace?: UnsubscribeFunc; | ||||
|   @state() private _urlPath?: string | null; | ||||
|  | ||||
|   private _urlPath?: string | null; | ||||
|   private _unsubLovelace?: UnsubscribeFunc; | ||||
|  | ||||
|   public processIncomingMessage(msg: HassMessage) { | ||||
|     if (msg.type === "connect") { | ||||
| @@ -68,8 +72,10 @@ export class HcMain extends HassElement { | ||||
|       !this._lovelaceConfig || | ||||
|       this._lovelacePath === null || | ||||
|       // Guard against part of HA not being loaded yet. | ||||
|       (this.hass && | ||||
|         (!this.hass.states || !this.hass.config || !this.hass.services)) | ||||
|       !this.hass || | ||||
|       !this.hass.states || | ||||
|       !this.hass.config || | ||||
|       !this.hass.services | ||||
|     ) { | ||||
|       return html` | ||||
|         <hc-launch-screen | ||||
| @@ -119,7 +125,7 @@ export class HcMain extends HassElement { | ||||
|  | ||||
|     if (this.hass) { | ||||
|       status.hassUrl = this.hass.auth.data.hassUrl; | ||||
|       status.lovelacePath = this._lovelacePath!; | ||||
|       status.lovelacePath = this._lovelacePath; | ||||
|       status.urlPath = this._urlPath; | ||||
|     } | ||||
|  | ||||
| @@ -132,6 +138,26 @@ export class HcMain extends HassElement { | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private _sendError( | ||||
|     error_code: number, | ||||
|     error_message: string, | ||||
|     senderId?: string | ||||
|   ) { | ||||
|     const error: ReceiverErrorMessage = { | ||||
|       type: "receiver_error", | ||||
|       error_code, | ||||
|       error_message, | ||||
|     }; | ||||
|  | ||||
|     if (senderId) { | ||||
|       this.sendMessage(senderId, error); | ||||
|     } else { | ||||
|       for (const sender of castContext.getSenders()) { | ||||
|         this.sendMessage(sender.id, error); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private _dialogClosed = () => { | ||||
|     document.body.setAttribute("style", "overflow-y: auto !important"); | ||||
|   }; | ||||
| @@ -154,14 +180,18 @@ export class HcMain extends HassElement { | ||||
|         }), | ||||
|       }); | ||||
|     } catch (err: any) { | ||||
|       this._error = this._getErrorMessage(err); | ||||
|       const errorMessage = this._getErrorMessage(err); | ||||
|       this._error = errorMessage; | ||||
|       this._sendError(err, errorMessage); | ||||
|       return; | ||||
|     } | ||||
|     let connection; | ||||
|     try { | ||||
|       connection = await createConnection({ auth }); | ||||
|     } catch (err: any) { | ||||
|       this._error = this._getErrorMessage(err); | ||||
|       const errorMessage = this._getErrorMessage(err); | ||||
|       this._error = errorMessage; | ||||
|       this._sendError(err, errorMessage); | ||||
|       return; | ||||
|     } | ||||
|     if (this.hass) { | ||||
| @@ -173,24 +203,29 @@ export class HcMain extends HassElement { | ||||
|   } | ||||
|  | ||||
|   private async _handleShowLovelaceMessage(msg: ShowLovelaceViewMessage) { | ||||
|     this._showDemo = false; | ||||
|     // We should not get this command before we are connected. | ||||
|     // Means a client got out of sync. Let's send status to them. | ||||
|     if (!this.hass) { | ||||
|       this._sendStatus(msg.senderId!); | ||||
|       this._error = "Cannot show Lovelace because we're not connected."; | ||||
|       this._sendError(ReceiverErrorCode.NOT_CONNECTED, this._error); | ||||
|       return; | ||||
|     } | ||||
|     this._error = undefined; | ||||
|     if (msg.urlPath === "lovelace") { | ||||
|       msg.urlPath = null; | ||||
|     } | ||||
|     this._lovelacePath = msg.viewPath; | ||||
|     if (!this._unsubLovelace || this._urlPath !== msg.urlPath) { | ||||
|       this._urlPath = msg.urlPath; | ||||
|       this._lovelaceConfig = undefined; | ||||
|       if (this._unsubLovelace) { | ||||
|         this._unsubLovelace(); | ||||
|       } | ||||
|       const llColl = atLeastVersion(this.hass.connection.haVersion, 0, 107) | ||||
|         ? getLovelaceCollection(this.hass!.connection, msg.urlPath) | ||||
|         : getLegacyLovelaceCollection(this.hass!.connection); | ||||
|         ? getLovelaceCollection(this.hass.connection, msg.urlPath) | ||||
|         : getLegacyLovelaceCollection(this.hass.connection); | ||||
|       // We first do a single refresh because we need to check if there is LL | ||||
|       // configuration. | ||||
|       try { | ||||
| @@ -199,8 +234,16 @@ export class HcMain extends HassElement { | ||||
|           this._handleNewLovelaceConfig(lovelaceConfig) | ||||
|         ); | ||||
|       } catch (err: any) { | ||||
|         if ( | ||||
|           atLeastVersion(this.hass.connection.haVersion, 0, 107) && | ||||
|           err.code !== "config_not_found" | ||||
|         ) { | ||||
|           // eslint-disable-next-line | ||||
|           console.log("Error fetching Lovelace configuration", err, msg); | ||||
|           this._error = `Error fetching Lovelace configuration: ${err.message}`; | ||||
|           this._sendError(ReceiverErrorCode.FETCH_CONFIG_FAILED, this._error); | ||||
|           return; | ||||
|         } | ||||
|         // Generate a Lovelace config. | ||||
|         this._unsubLovelace = () => undefined; | ||||
|         await this._generateLovelaceConfig(); | ||||
| @@ -215,8 +258,6 @@ export class HcMain extends HassElement { | ||||
|         loadLovelaceResources(resources, this.hass!.auth.data.hassUrl); | ||||
|       } | ||||
|     } | ||||
|     this._showDemo = false; | ||||
|     this._lovelacePath = msg.viewPath; | ||||
|  | ||||
|     this._sendStatus(); | ||||
|   } | ||||
| @@ -237,7 +278,7 @@ export class HcMain extends HassElement { | ||||
|   } | ||||
|  | ||||
|   private _handleNewLovelaceConfig(lovelaceConfig: LovelaceConfig) { | ||||
|     castContext.setApplicationState(lovelaceConfig.title!); | ||||
|     castContext.setApplicationState(lovelaceConfig.title || ""); | ||||
|     this._lovelaceConfig = lovelaceConfig; | ||||
|   } | ||||
|  | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -82,6 +82,9 @@ export const mockEnergy = (hass: MockHomeAssistant) => { | ||||
|     ], | ||||
|   })); | ||||
|   hass.mockWS("energy/info", () => ({ cost_sensors: [] })); | ||||
|   hass.mockWS("energy/fossil_energy_consumption", ({ period }) => ({ | ||||
|     start: period === "month" ? 250 : period === "day" ? 10 : 2, | ||||
|   })); | ||||
|   const todayString = format(startOfToday(), "yyyy-MM-dd"); | ||||
|   const tomorrowString = format(startOfTomorrow(), "yyyy-MM-dd"); | ||||
|   hass.mockWS( | ||||
|   | ||||
| @@ -1,4 +1,10 @@ | ||||
| import { addHours, differenceInHours, endOfDay } from "date-fns"; | ||||
| import { | ||||
|   addDays, | ||||
|   addHours, | ||||
|   addMonths, | ||||
|   differenceInHours, | ||||
|   endOfDay, | ||||
| } from "date-fns"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { StatisticValue } from "../../../src/data/history"; | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
| @@ -70,6 +76,7 @@ const generateMeanStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| ) => { | ||||
| @@ -84,6 +91,7 @@ const generateMeanStatistics = ( | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       end: currentDate.toISOString(), | ||||
|       mean, | ||||
|       min: mean - Math.random() * maxDiff, | ||||
|       max: mean + Math.random() * maxDiff, | ||||
| @@ -92,7 +100,12 @@ const generateMeanStatistics = ( | ||||
|       sum: null, | ||||
|     }); | ||||
|     lastVal = mean; | ||||
|     currentDate = addHours(currentDate, 1); | ||||
|     currentDate = | ||||
|       period === "day" | ||||
|         ? addDays(currentDate, 1) | ||||
|         : period === "month" | ||||
|         ? addMonths(currentDate, 1) | ||||
|         : addHours(currentDate, 1); | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
| @@ -101,6 +114,7 @@ const generateSumStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| ) => { | ||||
| @@ -115,6 +129,7 @@ const generateSumStatistics = ( | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       end: currentDate.toISOString(), | ||||
|       mean: null, | ||||
|       min: null, | ||||
|       max: null, | ||||
| @@ -122,7 +137,12 @@ const generateSumStatistics = ( | ||||
|       state: initValue + sum, | ||||
|       sum, | ||||
|     }); | ||||
|     currentDate = addHours(currentDate, 1); | ||||
|     currentDate = | ||||
|       period === "day" | ||||
|         ? addDays(currentDate, 1) | ||||
|         : period === "month" | ||||
|         ? addMonths(currentDate, 1) | ||||
|         : addHours(currentDate, 1); | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
| @@ -131,6 +151,7 @@ const generateCurvedStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   _period: "5minute" | "hour" | "day" | "month" = "hour", | ||||
|   initValue: number, | ||||
|   maxDiff: number, | ||||
|   metered: boolean | ||||
| @@ -149,6 +170,7 @@ const generateCurvedStatistics = ( | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       end: currentDate.toISOString(), | ||||
|       mean: null, | ||||
|       min: null, | ||||
|       max: null, | ||||
| @@ -167,11 +189,38 @@ const generateCurvedStatistics = ( | ||||
|  | ||||
| const statisticsFunctions: Record< | ||||
|   string, | ||||
|   (id: string, start: Date, end: Date) => StatisticValue[] | ||||
|   ( | ||||
|     id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period: "5minute" | "hour" | "day" | "month" | ||||
|   ) => StatisticValue[] | ||||
| > = { | ||||
|   "sensor.energy_consumption_tarif_1": (id: string, start: Date, end: Date) => { | ||||
|   "sensor.energy_consumption_tarif_1": ( | ||||
|     id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period = "hour" | ||||
|   ) => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         id, | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const morningEnd = new Date(start.getTime() + 10 * 60 * 60 * 1000); | ||||
|     const morningLow = generateSumStatistics(id, start, morningEnd, 0, 0.7); | ||||
|     const morningLow = generateSumStatistics( | ||||
|       id, | ||||
|       start, | ||||
|       morningEnd, | ||||
|       period, | ||||
|       0, | ||||
|       0.7 | ||||
|     ); | ||||
|     const eveningStart = new Date(start.getTime() + 20 * 60 * 60 * 1000); | ||||
|     const morningFinalVal = morningLow.length | ||||
|       ? morningLow[morningLow.length - 1].sum! | ||||
| @@ -180,6 +229,7 @@ const statisticsFunctions: Record< | ||||
|       id, | ||||
|       morningEnd, | ||||
|       eveningStart, | ||||
|       period, | ||||
|       morningFinalVal, | ||||
|       0 | ||||
|     ); | ||||
| @@ -187,39 +237,71 @@ const statisticsFunctions: Record< | ||||
|       id, | ||||
|       eveningStart, | ||||
|       end, | ||||
|       period, | ||||
|       morningFinalVal, | ||||
|       0.7 | ||||
|     ); | ||||
|     return [...morningLow, ...empty, ...eveningLow]; | ||||
|   }, | ||||
|   "sensor.energy_consumption_tarif_2": (id: string, start: Date, end: Date) => { | ||||
|   "sensor.energy_consumption_tarif_2": ( | ||||
|     id: string, | ||||
|     start: Date, | ||||
|     end: Date, | ||||
|     period = "hour" | ||||
|   ) => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         id, | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const morningEnd = new Date(start.getTime() + 9 * 60 * 60 * 1000); | ||||
|     const eveningStart = new Date(start.getTime() + 20 * 60 * 60 * 1000); | ||||
|     const highTarif = generateSumStatistics( | ||||
|       id, | ||||
|       morningEnd, | ||||
|       eveningStart, | ||||
|       period, | ||||
|       0, | ||||
|       0.3 | ||||
|     ); | ||||
|     const highTarifFinalVal = highTarif.length | ||||
|       ? highTarif[highTarif.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(id, start, morningEnd, 0, 0); | ||||
|     const morning = generateSumStatistics(id, start, morningEnd, period, 0, 0); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       eveningStart, | ||||
|       end, | ||||
|       period, | ||||
|       highTarifFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     return [...morning, ...highTarif, ...evening]; | ||||
|   }, | ||||
|   "sensor.energy_production_tarif_1": (id, start, end) => | ||||
|     generateSumStatistics(id, start, end, 0, 0), | ||||
|   "sensor.energy_production_tarif_1_compensation": (id, start, end) => | ||||
|     generateSumStatistics(id, start, end, 0, 0), | ||||
|   "sensor.energy_production_tarif_2": (id, start, end) => { | ||||
|   "sensor.energy_production_tarif_1": (id, start, end, period = "hour") => | ||||
|     generateSumStatistics(id, start, end, period, 0, 0), | ||||
|   "sensor.energy_production_tarif_1_compensation": ( | ||||
|     id, | ||||
|     start, | ||||
|     end, | ||||
|     period = "hour" | ||||
|   ) => generateSumStatistics(id, start, end, period, 0, 0), | ||||
|   "sensor.energy_production_tarif_2": (id, start, end, period = "hour") => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         id, | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const productionStart = new Date(start.getTime() + 9 * 60 * 60 * 1000); | ||||
|     const productionEnd = new Date(start.getTime() + 21 * 60 * 60 * 1000); | ||||
|     const dayEnd = new Date(endOfDay(productionEnd)); | ||||
| @@ -227,6 +309,7 @@ const statisticsFunctions: Record< | ||||
|       id, | ||||
|       productionStart, | ||||
|       productionEnd, | ||||
|       period, | ||||
|       0, | ||||
|       0.15, | ||||
|       true | ||||
| @@ -234,18 +317,43 @@ const statisticsFunctions: Record< | ||||
|     const productionFinalVal = production.length | ||||
|       ? production[production.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(id, start, productionStart, 0, 0); | ||||
|     const morning = generateSumStatistics( | ||||
|       id, | ||||
|       start, | ||||
|       productionStart, | ||||
|       period, | ||||
|       0, | ||||
|       0 | ||||
|     ); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       productionEnd, | ||||
|       dayEnd, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const rest = generateSumStatistics(id, dayEnd, end, productionFinalVal, 1); | ||||
|     const rest = generateSumStatistics( | ||||
|       id, | ||||
|       dayEnd, | ||||
|       end, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       1 | ||||
|     ); | ||||
|     return [...morning, ...production, ...evening, ...rest]; | ||||
|   }, | ||||
|   "sensor.solar_production": (id, start, end) => { | ||||
|   "sensor.solar_production": (id, start, end, period = "hour") => { | ||||
|     if (period !== "hour") { | ||||
|       return generateSumStatistics( | ||||
|         id, | ||||
|         start, | ||||
|         end, | ||||
|         period, | ||||
|         0, | ||||
|         period === "day" ? 17 : 504 | ||||
|       ); | ||||
|     } | ||||
|     const productionStart = new Date(start.getTime() + 7 * 60 * 60 * 1000); | ||||
|     const productionEnd = new Date(start.getTime() + 23 * 60 * 60 * 1000); | ||||
|     const dayEnd = new Date(endOfDay(productionEnd)); | ||||
| @@ -253,6 +361,7 @@ const statisticsFunctions: Record< | ||||
|       id, | ||||
|       productionStart, | ||||
|       productionEnd, | ||||
|       period, | ||||
|       0, | ||||
|       0.3, | ||||
|       true | ||||
| @@ -260,19 +369,32 @@ const statisticsFunctions: Record< | ||||
|     const productionFinalVal = production.length | ||||
|       ? production[production.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(id, start, productionStart, 0, 0); | ||||
|     const morning = generateSumStatistics( | ||||
|       id, | ||||
|       start, | ||||
|       productionStart, | ||||
|       period, | ||||
|       0, | ||||
|       0 | ||||
|     ); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       productionEnd, | ||||
|       dayEnd, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const rest = generateSumStatistics(id, dayEnd, end, productionFinalVal, 2); | ||||
|     const rest = generateSumStatistics( | ||||
|       id, | ||||
|       dayEnd, | ||||
|       end, | ||||
|       period, | ||||
|       productionFinalVal, | ||||
|       2 | ||||
|     ); | ||||
|     return [...morning, ...production, ...evening, ...rest]; | ||||
|   }, | ||||
|   "sensor.grid_fossil_fuel_percentage": (id, start, end) => | ||||
|     generateMeanStatistics(id, start, end, 35, 1.3), | ||||
| }; | ||||
|  | ||||
| export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
| @@ -347,7 +469,7 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|   mockHass.mockWS("history/list_statistic_ids", () => []); | ||||
|   mockHass.mockWS( | ||||
|     "history/statistics_during_period", | ||||
|     ({ statistic_ids, start_time, end_time }, hass) => { | ||||
|     ({ statistic_ids, start_time, end_time, period }, hass) => { | ||||
|       const start = new Date(start_time); | ||||
|       const end = end_time ? new Date(end_time) : new Date(); | ||||
|  | ||||
| @@ -355,7 +477,7 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|  | ||||
|       statistic_ids.forEach((id: string) => { | ||||
|         if (id in statisticsFunctions) { | ||||
|           statistics[id] = statisticsFunctions[id](id, start, end); | ||||
|           statistics[id] = statisticsFunctions[id](id, start, end, period); | ||||
|         } else { | ||||
|           const entityState = hass.states[id]; | ||||
|           const state = entityState ? Number(entityState.state) : 1; | ||||
| @@ -365,6 +487,7 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|                   id, | ||||
|                   start, | ||||
|                   end, | ||||
|                   period, | ||||
|                   state, | ||||
|                   state * (state > 80 ? 0.01 : 0.05) | ||||
|                 ) | ||||
| @@ -372,6 +495,7 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|                   id, | ||||
|                   start, | ||||
|                   end, | ||||
|                   period, | ||||
|                   state, | ||||
|                   state * (state > 80 ? 0.05 : 0.1) | ||||
|                 ); | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/office.jpg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/office.jpg
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 147 KiB | 
| @@ -1,6 +1,6 @@ | ||||
| #!/bin/bash | ||||
|  | ||||
| TARGET_LABEL="Needs gallery preview" | ||||
| TARGET_LABEL="needs design preview" | ||||
|  | ||||
| if [[ "$NETLIFY" != "true" ]]; then | ||||
|   echo "This script can only be run on Netlify" | ||||
| @@ -13,16 +13,14 @@ function createStatus() { | ||||
|   target_url="$3" | ||||
|   curl -X POST -H "Accept: application/vnd.github.v3+json" -H "Authorization: token $GITHUB_TOKEN" \ | ||||
|     "https://api.github.com/repos/home-assistant/frontend/statuses/$COMMIT_REF" \ | ||||
|     -d '{"state": "'"${state}"'", "context": "Netlify/Gallery Preview Build", "description": "'"$description"'", "target_url":  "'"$target_url"'"}' | ||||
|     -d '{"state": "'"${state}"'", "context": "Netlify/Design Preview Build", "description": "'"$description"'", "target_url":  "'"$target_url"'"}' | ||||
| } | ||||
|  | ||||
|  | ||||
| if [[ "${PULL_REQUEST}" == "false" ]]; then | ||||
|   gulp build-gallery | ||||
| else | ||||
| if [[ "${PULL_REQUEST}" == "true" ]]; then | ||||
|   if [[ "$(curl -sSLf -H "Accept: application/vnd.github.v3+json" -H "Authorization: token $GITHUB_TOKEN" \ | ||||
|     "https://api.github.com/repos/home-assistant/frontend/pulls/${REVIEW_ID}" | jq '.labels[].name' -r)" =~ "$TARGET_LABEL" ]]; then | ||||
|     createStatus "pending" "Building gallery preview" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||
|     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" | ||||
| @@ -32,4 +30,6 @@ else | ||||
|   else | ||||
|     createStatus "success" "Build was not requested by PR label" | ||||
|   fi | ||||
| elif [[ "$INCOMING_HOOK_BODY" == "NIGHTLY" ]]; then | ||||
|   gulp build-gallery | ||||
| fi | ||||
|   | ||||
							
								
								
									
										48
									
								
								gallery/sidebar.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								gallery/sidebar.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| module.exports = [ | ||||
|   { | ||||
|     // This section has no header and so all page links are shown directly in the sidebar | ||||
|     category: "concepts", | ||||
|     pages: ["home"], | ||||
|   }, | ||||
|  | ||||
|   { | ||||
|     category: "lovelace", | ||||
|     // Label for in the sidebar | ||||
|     header: "Lovelace", | ||||
|     // Specify order of pages. Any pages in the category folder but not listed here will | ||||
|     // automatically be added after the pages listed here. | ||||
|     pages: ["introduction"], | ||||
|   }, | ||||
|   { | ||||
|     category: "automation", | ||||
|     header: "Automation", | ||||
|     pages: [ | ||||
|       "editor-trigger", | ||||
|       "editor-condition", | ||||
|       "editor-action", | ||||
|       "selectors", | ||||
|       "trace", | ||||
|       "trace-timeline", | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     category: "components", | ||||
|     header: "Components", | ||||
|   }, | ||||
|   { | ||||
|     category: "more-info", | ||||
|     header: "More Info dialogs", | ||||
|   }, | ||||
|   { | ||||
|     category: "misc", | ||||
|     header: "Miscelaneous", | ||||
|   }, | ||||
|   { | ||||
|     category: "user-test", | ||||
|     header: "User Tests", | ||||
|   }, | ||||
|   { | ||||
|     category: "design.home-assistant.io", | ||||
|     header: "Design Documentation", | ||||
|   }, | ||||
| ]; | ||||
| @@ -3,6 +3,7 @@ import { html, LitElement, css, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element"; | ||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||
| import "../../../src/components/ha-card"; | ||||
|  | ||||
| @customElement("demo-black-white-row") | ||||
| class DemoBlackWhiteRow extends LitElement { | ||||
| @@ -52,17 +53,13 @@ class DemoBlackWhiteRow extends LitElement { | ||||
|  | ||||
|   firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     applyThemesOnElement( | ||||
|       this.shadowRoot!.querySelector(".dark"), | ||||
|       { | ||||
|     applyThemesOnElement(this.shadowRoot!.querySelector(".dark"), { | ||||
|       default_theme: "default", | ||||
|       default_dark_theme: "default", | ||||
|       themes: {}, | ||||
|         darkMode: false, | ||||
|       }, | ||||
|       "default", | ||||
|       { dark: true } | ||||
|     ); | ||||
|       darkMode: true, | ||||
|       theme: "default", | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   handleSubmit(ev) { | ||||
|   | ||||
| @@ -1,129 +0,0 @@ | ||||
| import { html } from "@polymer/polymer/lib/utils/html-tag"; | ||||
| /* eslint-plugin-disable lit */ | ||||
| import { PolymerElement } from "@polymer/polymer/polymer-element"; | ||||
| import { load } from "js-yaml"; | ||||
| import { createCardElement } from "../../../src/panels/lovelace/create-element/create-card-element"; | ||||
|  | ||||
| class DemoCard extends PolymerElement { | ||||
|   static get template() { | ||||
|     return html` | ||||
|       <style> | ||||
|         .root { | ||||
|           display: flex; | ||||
|         } | ||||
|         h2 { | ||||
|           margin: 0 0 20px; | ||||
|           color: var(--primary-color); | ||||
|         } | ||||
|         h2 small { | ||||
|           font-size: 0.5em; | ||||
|           color: var(--primary-text-color); | ||||
|         } | ||||
|         #card { | ||||
|           max-width: 400px; | ||||
|           width: 100vw; | ||||
|         } | ||||
|         pre { | ||||
|           width: 400px; | ||||
|           margin: 0 16px; | ||||
|           overflow: auto; | ||||
|           color: var(--primary-text-color); | ||||
|         } | ||||
|         @media only screen and (max-width: 800px) { | ||||
|           .root { | ||||
|             flex-direction: column; | ||||
|           } | ||||
|           pre { | ||||
|             margin: 16px 0; | ||||
|           } | ||||
|         } | ||||
|       </style> | ||||
|       <h2> | ||||
|         [[config.heading]] | ||||
|         <template is="dom-if" if="[[_size]]"> | ||||
|           <small>(size [[_size]])</small> | ||||
|         </template> | ||||
|       </h2> | ||||
|       <div class="root"> | ||||
|         <div id="card"></div> | ||||
|         <template is="dom-if" if="[[showConfig]]"> | ||||
|           <pre>[[_trim(config.config)]]</pre> | ||||
|         </template> | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get properties() { | ||||
|     return { | ||||
|       hass: { | ||||
|         type: Object, | ||||
|         observer: "_hassChanged", | ||||
|       }, | ||||
|       config: { | ||||
|         type: Object, | ||||
|         observer: "_configChanged", | ||||
|       }, | ||||
|       showConfig: Boolean, | ||||
|       _size: { | ||||
|         type: Number, | ||||
|       }, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   ready() { | ||||
|     super.ready(); | ||||
|   } | ||||
|  | ||||
|   _configChanged(config) { | ||||
|     const card = this.$.card; | ||||
|     while (card.lastChild) { | ||||
|       card.removeChild(card.lastChild); | ||||
|     } | ||||
|  | ||||
|     const el = this._createCardElement(load(config.config)[0]); | ||||
|     card.appendChild(el); | ||||
|     this._getSize(el); | ||||
|   } | ||||
|  | ||||
|   async _getSize(el) { | ||||
|     await customElements.whenDefined(el.localName); | ||||
|  | ||||
|     if (!("getCardSize" in el)) { | ||||
|       this._size = undefined; | ||||
|       return; | ||||
|     } | ||||
|     this._size = await el.getCardSize(); | ||||
|   } | ||||
|  | ||||
|   _createCardElement(cardConfig) { | ||||
|     const element = createCardElement(cardConfig); | ||||
|     if (this.hass) { | ||||
|       element.hass = this.hass; | ||||
|     } | ||||
|     element.addEventListener( | ||||
|       "ll-rebuild", | ||||
|       (ev) => { | ||||
|         ev.stopPropagation(); | ||||
|         this._rebuildCard(element, cardConfig); | ||||
|       }, | ||||
|       { once: true } | ||||
|     ); | ||||
|     return element; | ||||
|   } | ||||
|  | ||||
|   _rebuildCard(cardElToReplace, config) { | ||||
|     const newCardEl = this._createCardElement(config); | ||||
|     cardElToReplace.parentElement.replaceChild(newCardEl, cardElToReplace); | ||||
|   } | ||||
|  | ||||
|   _hassChanged(hass) { | ||||
|     const card = this.$.card.lastChild; | ||||
|     if (card) card.hass = hass; | ||||
|   } | ||||
|  | ||||
|   _trim(config) { | ||||
|     return config.trim(); | ||||
|   } | ||||
| } | ||||
|  | ||||
| customElements.define("demo-card", DemoCard); | ||||
							
								
								
									
										129
									
								
								gallery/src/components/demo-card.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								gallery/src/components/demo-card.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | ||||
| import { load } from "js-yaml"; | ||||
| import { html, css, LitElement, PropertyValues } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { createCardElement } from "../../../src/panels/lovelace/create-element/create-card-element"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
|  | ||||
| export interface DemoCardConfig { | ||||
|   heading: string; | ||||
|   config: string; | ||||
| } | ||||
|  | ||||
| @customElement("demo-card") | ||||
| class DemoCard extends LitElement { | ||||
|   @property() public hass!: HomeAssistant; | ||||
|  | ||||
|   @property() public config!: DemoCardConfig; | ||||
|  | ||||
|   @property() public showConfig = false; | ||||
|  | ||||
|   @state() private _size?: number; | ||||
|  | ||||
|   @query("#card") private _card!: HTMLElement; | ||||
|  | ||||
|   render() { | ||||
|     return html` | ||||
|       <h2> | ||||
|         ${this.config.heading} | ||||
|         ${this._size !== undefined | ||||
|           ? html`<small>(size ${this._size})</small>` | ||||
|           : ""} | ||||
|       </h2> | ||||
|       <div class="root"> | ||||
|         <div id="card"></div> | ||||
|         ${this.showConfig ? html`<pre>${this.config.config.trim()}</pre>` : ""} | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   updated(changedProps: PropertyValues) { | ||||
|     super.updated(changedProps); | ||||
|  | ||||
|     if (changedProps.has("config")) { | ||||
|       const card = this._card; | ||||
|       while (card.lastChild) { | ||||
|         card.removeChild(card.lastChild); | ||||
|       } | ||||
|  | ||||
|       const el = this._createCardElement((load(this.config.config) as any)[0]); | ||||
|       card.appendChild(el); | ||||
|       this._getSize(el); | ||||
|     } | ||||
|  | ||||
|     if (changedProps.has("hass")) { | ||||
|       const card = this._card.lastChild; | ||||
|       if (card) { | ||||
|         (card as any).hass = this.hass; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   async _getSize(el) { | ||||
|     await customElements.whenDefined(el.localName); | ||||
|  | ||||
|     if (!("getCardSize" in el)) { | ||||
|       this._size = undefined; | ||||
|       return; | ||||
|     } | ||||
|     this._size = await el.getCardSize(); | ||||
|   } | ||||
|  | ||||
|   _createCardElement(cardConfig) { | ||||
|     const element = createCardElement(cardConfig); | ||||
|     if (this.hass) { | ||||
|       element.hass = this.hass; | ||||
|     } | ||||
|     element.addEventListener( | ||||
|       "ll-rebuild", | ||||
|       (ev) => { | ||||
|         ev.stopPropagation(); | ||||
|         this._rebuildCard(element, cardConfig); | ||||
|       }, | ||||
|       { once: true } | ||||
|     ); | ||||
|     return element; | ||||
|   } | ||||
|  | ||||
|   _rebuildCard(cardElToReplace, config) { | ||||
|     const newCardEl = this._createCardElement(config); | ||||
|     cardElToReplace.parentElement.replaceChild(newCardEl, cardElToReplace); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     .root { | ||||
|       display: flex; | ||||
|     } | ||||
|     h2 { | ||||
|       margin: 0 0 20px; | ||||
|       color: var(--primary-color); | ||||
|     } | ||||
|     h2 small { | ||||
|       font-size: 0.5em; | ||||
|       color: var(--primary-text-color); | ||||
|     } | ||||
|     #card { | ||||
|       max-width: 400px; | ||||
|       width: 100vw; | ||||
|     } | ||||
|     pre { | ||||
|       width: 400px; | ||||
|       margin: 0 16px; | ||||
|       overflow: auto; | ||||
|       color: var(--primary-text-color); | ||||
|     } | ||||
|     @media only screen and (max-width: 800px) { | ||||
|       .root { | ||||
|         flex-direction: column; | ||||
|       } | ||||
|       pre { | ||||
|         margin: 16px 0; | ||||
|       } | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-card": DemoCard; | ||||
|   } | ||||
| } | ||||
| @@ -1,83 +0,0 @@ | ||||
| import "@polymer/app-layout/app-toolbar/app-toolbar"; | ||||
| import { html } from "@polymer/polymer/lib/utils/html-tag"; | ||||
| /* eslint-plugin-disable lit */ | ||||
| import { PolymerElement } from "@polymer/polymer/polymer-element"; | ||||
| import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element"; | ||||
| import "../../../src/components/ha-formfield"; | ||||
| import "../../../src/components/ha-switch"; | ||||
| import "./demo-card"; | ||||
|  | ||||
| class DemoCards extends PolymerElement { | ||||
|   static get template() { | ||||
|     return html` | ||||
|       <style> | ||||
|         #container { | ||||
|           min-height: calc(100vh - 128px); | ||||
|           background: var(--primary-background-color); | ||||
|         } | ||||
|         .cards { | ||||
|           display: flex; | ||||
|           flex-wrap: wrap; | ||||
|           justify-content: center; | ||||
|         } | ||||
|         demo-card { | ||||
|           margin: 16px 16px 32px; | ||||
|         } | ||||
|         app-toolbar { | ||||
|           background-color: var(--light-primary-color); | ||||
|         } | ||||
|         .filters { | ||||
|           margin-left: 60px; | ||||
|         } | ||||
|         ha-formfield { | ||||
|           margin-right: 16px; | ||||
|         } | ||||
|       </style> | ||||
|       <app-toolbar> | ||||
|         <div class="filters"> | ||||
|           <ha-formfield label="Show config"> | ||||
|             <ha-switch checked="[[_showConfig]]" on-change="_showConfigToggled"> | ||||
|             </ha-switch> | ||||
|           </ha-formfield> | ||||
|           <ha-formfield label="Dark theme"> | ||||
|             <ha-switch on-change="_darkThemeToggled"> </ha-switch> | ||||
|           </ha-formfield> | ||||
|         </div> | ||||
|       </app-toolbar> | ||||
|       <div id="container"> | ||||
|         <div class="cards"> | ||||
|           <template is="dom-repeat" items="[[configs]]"> | ||||
|             <demo-card | ||||
|               config="[[item]]" | ||||
|               show-config="[[_showConfig]]" | ||||
|               hass="[[hass]]" | ||||
|             ></demo-card> | ||||
|           </template> | ||||
|         </div> | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get properties() { | ||||
|     return { | ||||
|       configs: Object, | ||||
|       hass: Object, | ||||
|       _showConfig: { | ||||
|         type: Boolean, | ||||
|         value: false, | ||||
|       }, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   _showConfigToggled(ev) { | ||||
|     this._showConfig = ev.target.checked; | ||||
|   } | ||||
|  | ||||
|   _darkThemeToggled(ev) { | ||||
|     applyThemesOnElement(this.$.container, { themes: {} }, "default", { | ||||
|       dark: ev.target.checked, | ||||
|     }); | ||||
|   } | ||||
| } | ||||
|  | ||||
| customElements.define("demo-cards", DemoCards); | ||||
							
								
								
									
										88
									
								
								gallery/src/components/demo-cards.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								gallery/src/components/demo-cards.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | ||||
| import "@polymer/app-layout/app-toolbar/app-toolbar"; | ||||
| import { html, css, LitElement } from "lit"; | ||||
| import { customElement, property, query, state } from "lit/decorators"; | ||||
| import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element"; | ||||
| import "../../../src/components/ha-formfield"; | ||||
| import "../../../src/components/ha-switch"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
| import "./demo-card"; | ||||
| import type { DemoCardConfig } from "./demo-card"; | ||||
|  | ||||
| @customElement("demo-cards") | ||||
| class DemoCards extends LitElement { | ||||
|   @property() public configs!: DemoCardConfig[]; | ||||
|  | ||||
|   @property() public hass!: HomeAssistant; | ||||
|  | ||||
|   @state() private _showConfig = false; | ||||
|  | ||||
|   @query("#container") private _container!: HTMLElement; | ||||
|  | ||||
|   render() { | ||||
|     return html` | ||||
|       <app-toolbar> | ||||
|         <div class="filters"> | ||||
|           <ha-formfield label="Show config"> | ||||
|             <ha-switch | ||||
|               .checked=${this._showConfig} | ||||
|               @change=${this._showConfigToggled} | ||||
|             > | ||||
|             </ha-switch> | ||||
|           </ha-formfield> | ||||
|           <ha-formfield label="Dark theme"> | ||||
|             <ha-switch @change=${this._darkThemeToggled}> </ha-switch> | ||||
|           </ha-formfield> | ||||
|         </div> | ||||
|       </app-toolbar> | ||||
|       <div id="container"> | ||||
|         <div class="cards"> | ||||
|           ${this.configs.map( | ||||
|             (config) => html` | ||||
|               <demo-card | ||||
|                 .config=${config} | ||||
|                 .showConfig=${this._showConfig} | ||||
|                 .hass=${this.hass} | ||||
|               ></demo-card> | ||||
|             ` | ||||
|           )} | ||||
|         </div> | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   _showConfigToggled(ev) { | ||||
|     this._showConfig = ev.target.checked; | ||||
|   } | ||||
|  | ||||
|   _darkThemeToggled(ev) { | ||||
|     applyThemesOnElement(this._container, { themes: {} } as any, "default", { | ||||
|       dark: ev.target.checked, | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   static styles = css` | ||||
|     .cards { | ||||
|       display: flex; | ||||
|       flex-wrap: wrap; | ||||
|       justify-content: center; | ||||
|     } | ||||
|     demo-card { | ||||
|       margin: 16px 16px 32px; | ||||
|     } | ||||
|     app-toolbar { | ||||
|       background-color: var(--light-primary-color); | ||||
|     } | ||||
|     .filters { | ||||
|       margin-left: 60px; | ||||
|     } | ||||
|     ha-formfield { | ||||
|       margin-right: 16px; | ||||
|     } | ||||
|   `; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-cards": DemoCards; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										46
									
								
								gallery/src/components/page-description.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								gallery/src/components/page-description.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| import { html, css } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import { until } from "lit/directives/until"; | ||||
| import { HaMarkdown } from "../../../src/components/ha-markdown"; | ||||
| import { PAGES } from "../../build/import-pages"; | ||||
|  | ||||
| @customElement("page-description") | ||||
| class PageDescription extends HaMarkdown { | ||||
|   @property() public page!: string; | ||||
|  | ||||
|   render() { | ||||
|     if (!PAGES[this.page].description) { | ||||
|       return html``; | ||||
|     } | ||||
|     return html` | ||||
|       ${until( | ||||
|         PAGES[this.page] | ||||
|           .description() | ||||
|           .then((content) => html`<div class="root">${content}</div>`), | ||||
|         "" | ||||
|       )} | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static styles = [ | ||||
|     HaMarkdown.styles, | ||||
|     css` | ||||
|       .root { | ||||
|         max-width: 800px; | ||||
|         margin: 0 auto; | ||||
|       } | ||||
|       .root > *:first-child { | ||||
|         margin-top: 0; | ||||
|       } | ||||
|       .root > *:last-child { | ||||
|         margin-bottom: 0; | ||||
|       } | ||||
|     `, | ||||
|   ]; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "page-description": PageDescription; | ||||
|   } | ||||
| } | ||||
| @@ -1,230 +0,0 @@ | ||||
| import "@polymer/app-layout/app-header-layout/app-header-layout"; | ||||
| import "@polymer/app-layout/app-header/app-header"; | ||||
| import "@polymer/app-layout/app-toolbar/app-toolbar"; | ||||
| import "@polymer/paper-item/paper-item"; | ||||
| import "@polymer/paper-item/paper-item-body"; | ||||
| import { html } from "@polymer/polymer/lib/utils/html-tag"; | ||||
| /* eslint-plugin-disable lit */ | ||||
| import { PolymerElement } from "@polymer/polymer/polymer-element"; | ||||
| import "../../src/components/ha-card"; | ||||
| import "../../src/components/ha-icon"; | ||||
| import "../../src/components/ha-icon-button"; | ||||
| import "../../src/managers/notification-manager"; | ||||
| import "../../src/styles/polymer-ha-style"; | ||||
| // eslint-disable-next-line import/extensions | ||||
| import { DEMOS } from "../build/import-demos"; | ||||
|  | ||||
| class HaGallery extends PolymerElement { | ||||
|   static get template() { | ||||
|     return html` | ||||
|       <style include="iron-positioning ha-style"> | ||||
|         :host { | ||||
|           -ms-user-select: initial; | ||||
|           -webkit-user-select: initial; | ||||
|           -moz-user-select: initial; | ||||
|         } | ||||
|         app-header-layout { | ||||
|           min-height: 100vh; | ||||
|         } | ||||
|         ha-icon-button.invisible { | ||||
|           visibility: hidden; | ||||
|         } | ||||
|  | ||||
|         .pickers { | ||||
|           display: flex; | ||||
|           flex-wrap: wrap; | ||||
|           justify-content: center; | ||||
|           align-items: start; | ||||
|         } | ||||
|  | ||||
|         .pickers ha-card { | ||||
|           width: 400px; | ||||
|           display: block; | ||||
|           margin: 16px 8px; | ||||
|         } | ||||
|  | ||||
|         .pickers ha-card:last-child { | ||||
|           margin-bottom: 16px; | ||||
|         } | ||||
|  | ||||
|         .intro { | ||||
|           margin: -1em 0; | ||||
|         } | ||||
|  | ||||
|         p a { | ||||
|           color: var(--primary-color); | ||||
|         } | ||||
|  | ||||
|         a { | ||||
|           color: var(--primary-text-color); | ||||
|           text-decoration: none; | ||||
|         } | ||||
|       </style> | ||||
|  | ||||
|       <app-header-layout> | ||||
|         <app-header slot="header" fixed> | ||||
|           <app-toolbar> | ||||
|             <ha-icon-button | ||||
|               on-click="_backTapped" | ||||
|               class$="[[_computeHeaderButtonClass(_demo)]]" | ||||
|             > | ||||
|               <ha-icon icon="hass:arrow-left"></ha-icon> | ||||
|             </ha-icon-button> | ||||
|             <div main-title> | ||||
|               [[_withDefault(_demo, "Home Assistant Gallery")]] | ||||
|             </div> | ||||
|           </app-toolbar> | ||||
|         </app-header> | ||||
|  | ||||
|         <div class="content"> | ||||
|           <div id="demo"></div> | ||||
|           <template is="dom-if" if="[[!_demo]]"> | ||||
|             <div class="pickers"> | ||||
|               <ha-card header="Lovelace Card Demos"> | ||||
|                 <div class="card-content intro"> | ||||
|                   <p> | ||||
|                     Lovelace has many different cards. Each card allows the user | ||||
|                     to tell a different story about what is going on in their | ||||
|                     house. These cards are very customizable, as no household is | ||||
|                     the same. | ||||
|                   </p> | ||||
|  | ||||
|                   <p> | ||||
|                     This gallery helps our developers and designers to see all | ||||
|                     the different states that each card can be in. | ||||
|                   </p> | ||||
|  | ||||
|                   <p> | ||||
|                     Check | ||||
|                     <a href="https://www.home-assistant.io/lovelace" | ||||
|                       >the official website</a | ||||
|                     > | ||||
|                     for instructions on how to get started with Lovelace. | ||||
|                   </p> | ||||
|                 </div> | ||||
|                 <template is="dom-repeat" items="[[_lovelaceDemos]]"> | ||||
|                   <a href="#[[item]]"> | ||||
|                     <paper-item> | ||||
|                       <paper-item-body>{{ item }}</paper-item-body> | ||||
|                       <ha-icon icon="hass:chevron-right"></ha-icon> | ||||
|                     </paper-item> | ||||
|                   </a> | ||||
|                 </template> | ||||
|               </ha-card> | ||||
|  | ||||
|               <ha-card header="Other Demos"> | ||||
|                 <div class="card-content intro"></div> | ||||
|                 <template is="dom-repeat" items="[[_restDemos]]"> | ||||
|                   <a href="#[[item]]"> | ||||
|                     <paper-item> | ||||
|                       <paper-item-body>{{ item }}</paper-item-body> | ||||
|                       <ha-icon icon="hass:chevron-right"></ha-icon> | ||||
|                     </paper-item> | ||||
|                   </a> | ||||
|                 </template> | ||||
|               </ha-card> | ||||
|             </div> | ||||
|           </template> | ||||
|         </div> | ||||
|       </app-header-layout> | ||||
|       <notification-manager | ||||
|         hass="[[_fakeHass]]" | ||||
|         id="notifications" | ||||
|       ></notification-manager> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get properties() { | ||||
|     return { | ||||
|       _fakeHass: { | ||||
|         type: Object, | ||||
|         // Just enough for computeRTL | ||||
|         value: { | ||||
|           language: "en", | ||||
|           translationMetadata: { | ||||
|             translations: {}, | ||||
|           }, | ||||
|         }, | ||||
|       }, | ||||
|       _demo: { | ||||
|         type: String, | ||||
|         value: document.location.hash.substr(1), | ||||
|         observer: "_demoChanged", | ||||
|       }, | ||||
|       _demos: { | ||||
|         type: Array, | ||||
|         value: Object.keys(DEMOS), | ||||
|       }, | ||||
|       _lovelaceDemos: { | ||||
|         type: Array, | ||||
|         computed: "_computeLovelace(_demos)", | ||||
|       }, | ||||
|       _restDemos: { | ||||
|         type: Array, | ||||
|         computed: "_computeRest(_demos)", | ||||
|       }, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   ready() { | ||||
|     super.ready(); | ||||
|  | ||||
|     this.addEventListener("show-notification", (ev) => | ||||
|       this.$.notifications.showDialog({ message: ev.detail.message }) | ||||
|     ); | ||||
|  | ||||
|     this.addEventListener("alert-dismissed-clicked", () => | ||||
|       this.$.notifications.showDialog({ message: "Alert dismissed clicked" }) | ||||
|     ); | ||||
|  | ||||
|     this.addEventListener("alert-action-clicked", () => | ||||
|       this.$.notifications.showDialog({ message: "Alert action clicked" }) | ||||
|     ); | ||||
|  | ||||
|     this.addEventListener("hass-more-info", (ev) => { | ||||
|       if (ev.detail.entityId) { | ||||
|         this.$.notifications.showDialog({ | ||||
|           message: `Showing more info for ${ev.detail.entityId}`, | ||||
|         }); | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     window.addEventListener("hashchange", () => { | ||||
|       this._demo = document.location.hash.substr(1); | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   _withDefault(value, def) { | ||||
|     return value || def; | ||||
|   } | ||||
|  | ||||
|   _demoChanged(demo) { | ||||
|     const root = this.$.demo; | ||||
|  | ||||
|     while (root.lastChild) root.removeChild(root.lastChild); | ||||
|  | ||||
|     if (demo) { | ||||
|       DEMOS[demo](); | ||||
|       const el = document.createElement(demo); | ||||
|       root.appendChild(el); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   _computeHeaderButtonClass(demo) { | ||||
|     return demo ? "" : "invisible"; | ||||
|   } | ||||
|  | ||||
|   _backTapped() { | ||||
|     document.location.hash = ""; | ||||
|   } | ||||
|  | ||||
|   _computeLovelace(demos) { | ||||
|     return demos.filter((demo) => demo.includes("hui")); | ||||
|   } | ||||
|  | ||||
|   _computeRest(demos) { | ||||
|     return demos.filter((demo) => !demo.includes("hui")); | ||||
|   } | ||||
| } | ||||
|  | ||||
| customElements.define("ha-gallery", HaGallery); | ||||
							
								
								
									
										257
									
								
								gallery/src/ha-gallery.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										257
									
								
								gallery/src/ha-gallery.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,257 @@ | ||||
| import { mdiMenu } from "@mdi/js"; | ||||
| import "@material/mwc-drawer"; | ||||
| import "@material/mwc-top-app-bar-fixed"; | ||||
| 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 { haStyle } from "../../src/resources/styles"; | ||||
| import { PAGES, SIDEBAR } from "../build/import-pages"; | ||||
| import { dynamicElement } from "../../src/common/dom/dynamic-element-directive"; | ||||
| import "./components/page-description"; | ||||
|  | ||||
| const GITHUB_DEMO_URL = | ||||
|   "https://github.com/home-assistant/frontend/blob/dev/gallery/src/pages/"; | ||||
|  | ||||
| const FAKE_HASS = { | ||||
|   // Just enough for computeRTL for notification-manager | ||||
|   language: "en", | ||||
|   translationMetadata: { | ||||
|     translations: {}, | ||||
|   }, | ||||
| }; | ||||
|  | ||||
| @customElement("ha-gallery") | ||||
| class HaGallery extends LitElement { | ||||
|   @property() private _page = | ||||
|     document.location.hash.substring(1) || | ||||
|     `${SIDEBAR[0].category}/${SIDEBAR[0].pages![0]}`; | ||||
|  | ||||
|   @query("notification-manager") | ||||
|   private _notifications!: HTMLElementTagNameMap["notification-manager"]; | ||||
|  | ||||
|   @query("mwc-drawer") | ||||
|   private _drawer!: HTMLElementTagNameMap["mwc-drawer"]; | ||||
|  | ||||
|   private _narrow = window.matchMedia("(max-width: 600px)").matches; | ||||
|  | ||||
|   render() { | ||||
|     const sidebar: unknown[] = []; | ||||
|  | ||||
|     for (const group of SIDEBAR) { | ||||
|       const links: unknown[] = []; | ||||
|  | ||||
|       for (const page of group.pages!) { | ||||
|         const key = `${group.category}/${page}`; | ||||
|         const active = this._page === key; | ||||
|         const title = PAGES[key].metadata.title || page; | ||||
|         links.push(html` | ||||
|           <a ?active=${active} href=${`#${group.category}/${page}`}>${title}</a> | ||||
|         `); | ||||
|       } | ||||
|  | ||||
|       sidebar.push( | ||||
|         group.header | ||||
|           ? html` | ||||
|               <details> | ||||
|                 <summary class="section">${group.header}</summary> | ||||
|                 ${links} | ||||
|               </details> | ||||
|             ` | ||||
|           : links | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return html` | ||||
|       <mwc-drawer | ||||
|         hasHeader | ||||
|         .open=${!this._narrow} | ||||
|         .type=${this._narrow ? "modal" : "dismissible"} | ||||
|       > | ||||
|         <span slot="title">Home Assistant Design</span> | ||||
|         <!-- <span slot="subtitle">subtitle</span> --> | ||||
|         <div class="sidebar">${sidebar}</div> | ||||
|         <div slot="appContent"> | ||||
|           <mwc-top-app-bar-fixed> | ||||
|             <ha-icon-button | ||||
|               slot="navigationIcon" | ||||
|               @click=${this._menuTapped} | ||||
|               .path=${mdiMenu} | ||||
|             ></ha-icon-button> | ||||
|  | ||||
|             <div slot="title"> | ||||
|               ${PAGES[this._page].metadata.title || this._page.split("/")[1]} | ||||
|             </div> | ||||
|           </mwc-top-app-bar-fixed> | ||||
|           <div class="content"> | ||||
|             ${PAGES[this._page].description | ||||
|               ? html` | ||||
|                   <page-description .page=${this._page}></page-description> | ||||
|                 ` | ||||
|               : ""} | ||||
|             ${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> | ||||
|         </div> | ||||
|       </mwc-drawer> | ||||
|       <notification-manager | ||||
|         .hass=${FAKE_HASS} | ||||
|         id="notifications" | ||||
|       ></notification-manager> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   firstUpdated(changedProps: PropertyValues) { | ||||
|     super.firstUpdated(changedProps); | ||||
|  | ||||
|     this.addEventListener("show-notification", (ev) => | ||||
|       this._notifications.showDialog({ message: ev.detail.message }) | ||||
|     ); | ||||
|     this.addEventListener("alert-dismissed-clicked", () => | ||||
|       this._notifications.showDialog({ message: "Alert dismissed clicked" }) | ||||
|     ); | ||||
|     this.addEventListener("hass-more-info", (ev) => { | ||||
|       if (ev.detail.entityId) { | ||||
|         this._notifications.showDialog({ | ||||
|           message: `Showing more info for ${ev.detail.entityId}`, | ||||
|         }); | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     document.location.hash = this._page; | ||||
|  | ||||
|     window.addEventListener("hashchange", () => { | ||||
|       this._page = document.location.hash.substring(1); | ||||
|       if (this._narrow) { | ||||
|         this._drawer.open = false; | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
|  | ||||
|   updated(changedProps: PropertyValues) { | ||||
|     super.updated(changedProps); | ||||
|     if (!changedProps.has("_page")) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     if (PAGES[this._page].demo) { | ||||
|       PAGES[this._page].demo(); | ||||
|     } | ||||
|  | ||||
|     const menuItem = this.shadowRoot!.querySelector( | ||||
|       `a[href="#${this._page}"]` | ||||
|     )!; | ||||
|     // Make sure section is expanded | ||||
|     if (menuItem.parentElement instanceof HTMLDetailsElement) { | ||||
|       menuItem.parentElement.open = true; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   _menuTapped() { | ||||
|     this._drawer.open = !this._drawer.open; | ||||
|   } | ||||
|  | ||||
|   static styles = [ | ||||
|     haStyle, | ||||
|     css` | ||||
|       :host { | ||||
|         -ms-user-select: initial; | ||||
|         -webkit-user-select: initial; | ||||
|         -moz-user-select: initial; | ||||
|       } | ||||
|  | ||||
|       .sidebar { | ||||
|         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; | ||||
|         text-decoration: none; | ||||
|         position: relative; | ||||
|       } | ||||
|  | ||||
|       .sidebar a[active]::before { | ||||
|         border-radius: 4px; | ||||
|         position: absolute; | ||||
|         top: 0; | ||||
|         right: 2px; | ||||
|         bottom: 0; | ||||
|         left: 2px; | ||||
|         pointer-events: none; | ||||
|         content: ""; | ||||
|         transition: opacity 15ms linear; | ||||
|         will-change: opacity; | ||||
|         background-color: var(--sidebar-selected-icon-color); | ||||
|         opacity: 0.12; | ||||
|       } | ||||
|  | ||||
|       div[slot="appContent"] { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         min-height: 100vh; | ||||
|         background: var(--primary-background-color); | ||||
|       } | ||||
|  | ||||
|       .content { | ||||
|         flex: 1; | ||||
|       } | ||||
|  | ||||
|       page-description { | ||||
|         margin: 16px; | ||||
|       } | ||||
|  | ||||
|       .page-footer { | ||||
|         text-align: center; | ||||
|         margin: 16px 0; | ||||
|         padding-top: 16px; | ||||
|         border-top: 1px solid rgba(0, 0, 0, 0.12); | ||||
|       } | ||||
|  | ||||
|       .page-footer a { | ||||
|         display: inline-block; | ||||
|         margin: 0 8px; | ||||
|       } | ||||
|     `, | ||||
|   ]; | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "ha-gallery": HaGallery; | ||||
|   } | ||||
| } | ||||
| @@ -7,7 +7,7 @@ | ||||
|       content="width=device-width, initial-scale=1, shrink-to-fit=no" | ||||
|     /> | ||||
|     <meta name="theme-color" content="#2157BC" /> | ||||
|     <title>HAGallery</title> | ||||
|     <title>Home Assistant Design</title> | ||||
|  | ||||
|     <script type="module" src="<%= latestGalleryJS %>"></script> | ||||
|     <style> | ||||
|   | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/describe-action.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/describe-action.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Describe Action | ||||
| --- | ||||
| @@ -1,10 +1,10 @@ | ||||
| import { dump } from "js-yaml"; | ||||
| 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 { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import { describeAction } from "../../../../src/data/script_i18n"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| 
 | ||||
| const actions = [ | ||||
|   { wait_template: "{{ true }}", alias: "Something with an alias" }, | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/describe-condition.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/describe-condition.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Describe Condition | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { dump } from "js-yaml"; | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import { describeCondition } from "../../../src/data/automation_i18n"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import { describeCondition } from "../../../../src/data/automation_i18n"; | ||||
| 
 | ||||
| const conditions = [ | ||||
|   { condition: "and" }, | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/describe-trigger.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/describe-trigger.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Describe Trigger | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { dump } from "js-yaml"; | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import { describeTrigger } from "../../../src/data/automation_i18n"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import { describeTrigger } from "../../../../src/data/automation_i18n"; | ||||
| 
 | ||||
| const triggers = [ | ||||
|   { platform: "state" }, | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/editor-action.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/editor-action.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Actions | ||||
| --- | ||||
| @@ -1,25 +1,25 @@ | ||||
| /* 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/action/ha-automation-action"; | ||||
| import { HaChooseAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-choose"; | ||||
| import { HaDelayAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-delay"; | ||||
| 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 { 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"; | ||||
| import { Action } from "../../../src/data/script"; | ||||
| import { HaConditionAction } from "../../../src/panels/config/automation/action/types/ha-automation-action-condition"; | ||||
| 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/action/ha-automation-action"; | ||||
| import { HaChooseAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-choose"; | ||||
| import { HaDelayAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-delay"; | ||||
| 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 { 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"; | ||||
| import { Action } from "../../../../src/data/script"; | ||||
| import { HaConditionAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-condition"; | ||||
| 
 | ||||
| const SCHEMAS: { name: string; actions: Action[] }[] = [ | ||||
|   { name: "Event", actions: [HaEventAction.defaultConfig] }, | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/editor-condition.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/editor-condition.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Conditions | ||||
| --- | ||||
| @@ -1,24 +1,24 @@ | ||||
| /* 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 type { Condition } from "../../../src/data/automation"; | ||||
| import "../../../src/panels/config/automation/condition/ha-automation-condition"; | ||||
| import { HaDeviceCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-device"; | ||||
| import { HaLogicalCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-logical"; | ||||
| import HaNumericStateCondition from "../../../src/panels/config/automation/condition/types/ha-automation-condition-numeric_state"; | ||||
| import { HaStateCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-state"; | ||||
| import { HaSunCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-sun"; | ||||
| import { HaTemplateCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-template"; | ||||
| import { HaTimeCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-time"; | ||||
| import { HaTriggerCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-trigger"; | ||||
| import { HaZoneCondition } from "../../../src/panels/config/automation/condition/types/ha-automation-condition-zone"; | ||||
| 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 type { Condition } from "../../../../src/data/automation"; | ||||
| import "../../../../src/panels/config/automation/condition/ha-automation-condition"; | ||||
| import { HaDeviceCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-device"; | ||||
| import { HaLogicalCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-logical"; | ||||
| import HaNumericStateCondition from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-numeric_state"; | ||||
| import { HaStateCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-state"; | ||||
| import { HaSunCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-sun"; | ||||
| import { HaTemplateCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-template"; | ||||
| import { HaTimeCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-time"; | ||||
| import { HaTriggerCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-trigger"; | ||||
| import { HaZoneCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-zone"; | ||||
| 
 | ||||
| const SCHEMAS: { name: string; conditions: Condition[] }[] = [ | ||||
|   { | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/editor-trigger.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/editor-trigger.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Triggers | ||||
| --- | ||||
| @@ -1,29 +1,29 @@ | ||||
| /* 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 type { Trigger } from "../../../src/data/automation"; | ||||
| import { HaGeolocationTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-geo_location"; | ||||
| import { HaEventTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-event"; | ||||
| import { HaHassTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-homeassistant"; | ||||
| import { HaNumericStateTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-numeric_state"; | ||||
| import { HaSunTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-sun"; | ||||
| import { HaTagTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-tag"; | ||||
| import { HaTemplateTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-template"; | ||||
| import { HaTimeTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-time"; | ||||
| import { HaTimePatternTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-time_pattern"; | ||||
| import { HaWebhookTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-webhook"; | ||||
| import { HaZoneTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-zone"; | ||||
| import { HaDeviceTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-device"; | ||||
| import { HaStateTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-state"; | ||||
| import { HaMQTTTrigger } from "../../../src/panels/config/automation/trigger/types/ha-automation-trigger-mqtt"; | ||||
| import "../../../src/panels/config/automation/trigger/ha-automation-trigger"; | ||||
| 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 type { Trigger } from "../../../../src/data/automation"; | ||||
| import { HaGeolocationTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-geo_location"; | ||||
| import { HaEventTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-event"; | ||||
| import { HaHassTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-homeassistant"; | ||||
| import { HaNumericStateTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-numeric_state"; | ||||
| import { HaSunTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-sun"; | ||||
| import { HaTagTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-tag"; | ||||
| import { HaTemplateTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-template"; | ||||
| import { HaTimeTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-time"; | ||||
| import { HaTimePatternTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-time_pattern"; | ||||
| import { HaWebhookTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-webhook"; | ||||
| import { HaZoneTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-zone"; | ||||
| import { HaDeviceTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-device"; | ||||
| import { HaStateTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-state"; | ||||
| import { HaMQTTTrigger } from "../../../../src/panels/config/automation/trigger/types/ha-automation-trigger-mqtt"; | ||||
| import "../../../../src/panels/config/automation/trigger/ha-automation-trigger"; | ||||
| 
 | ||||
| const SCHEMAS: { name: string; triggers: Trigger[] }[] = [ | ||||
|   { | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/selectors.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/selectors.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Selectors | ||||
| --- | ||||
							
								
								
									
										102
									
								
								gallery/src/pages/automation/selectors.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										102
									
								
								gallery/src/pages/automation/selectors.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,102 @@ | ||||
| /* 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; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/trace-timeline.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/trace-timeline.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Trace Timelines | ||||
| --- | ||||
| @@ -1,13 +1,13 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/trace/hat-script-graph"; | ||||
| import "../../../src/components/trace/hat-trace-timeline"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
| import { mockDemoTrace } from "../data/traces/mock-demo-trace"; | ||||
| import { DemoTrace } from "../data/traces/types"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/trace/hat-script-graph"; | ||||
| import "../../../../src/components/trace/hat-trace-timeline"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import { mockDemoTrace } from "../../data/traces/mock-demo-trace"; | ||||
| import { DemoTrace } from "../../data/traces/types"; | ||||
| 
 | ||||
| const traces: DemoTrace[] = [ | ||||
|   mockDemoTrace({ state: "running" }), | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/automation/trace.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/automation/trace.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Trace Graphs | ||||
| --- | ||||
| @@ -1,14 +1,14 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/trace/hat-script-graph"; | ||||
| import "../../../src/components/trace/hat-trace-timeline"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/trace/hat-script-graph"; | ||||
| import "../../../../src/components/trace/hat-trace-timeline"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
| import { DemoTrace } from "../data/traces/types"; | ||||
| import { basicTrace } from "../data/traces/basic_trace"; | ||||
| import { motionLightTrace } from "../data/traces/motion-light-trace"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../../src/types"; | ||||
| import { DemoTrace } from "../../data/traces/types"; | ||||
| import { basicTrace } from "../../data/traces/basic_trace"; | ||||
| import { motionLightTrace } from "../../data/traces/motion-light-trace"; | ||||
| 
 | ||||
| const traces: DemoTrace[] = [basicTrace, motionLightTrace]; | ||||
| 
 | ||||
							
								
								
									
										181
									
								
								gallery/src/pages/components/ha-alert.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										181
									
								
								gallery/src/pages/components/ha-alert.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,181 @@ | ||||
| --- | ||||
| title: Alerts | ||||
| --- | ||||
|  | ||||
| # Alert `<ha-alert>` | ||||
| The alert offers four severity levels that set a distinctive icon and color. | ||||
|  | ||||
| <ha-alert alert-type="error"> | ||||
|   This is an error alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="warning"> | ||||
|   This is an warning alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="info"> | ||||
|   This is an info alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="success"> | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| **Note:** This component is by <a href="https://mui.com/components/alert/" rel="noopener noreferrer" target="_blank">MUI</a> and is not documented in the <a href="https://material.io" rel="noopener noreferrer" target="_blank">Material Design guidelines</a>. | ||||
|  | ||||
| 1. [Guidelines](#guidelines) | ||||
| 2. [Implementation](#implementation) | ||||
|  | ||||
| ### Resources | ||||
| | Type           | Link                             | Status    | | ||||
| |----------------|----------------------------------|-----------| | ||||
| | Design         | <a href="https://www.figma.com/community/file/967153512097289521/Home-Assistant-DesignKit" rel="noopener noreferrer" target="_blank">Home Assistant DesignKit</a> (Figma) | Available | | ||||
| | Implementation | <a href="https://github.com/home-assistant/frontend/blob/dev/src/components/ha-alert.ts" rel="noopener noreferrer" target="_blank">Web Component</a> (GitHub)            | Available | | ||||
|  | ||||
| ## Guidelines | ||||
| ### Usage | ||||
| An alert displays a short, important message in a way that attracts the user's attention without interrupting the user's task. | ||||
|  | ||||
| ### Anatomy | ||||
| *Documentation coming soon* | ||||
|  | ||||
| ### Error alert | ||||
| Error alerts | ||||
| *Real world example coming soon* | ||||
|  | ||||
| ### Warning alert | ||||
| Warning alerts | ||||
| *Real world example coming soon* | ||||
|  | ||||
| ### Info alert | ||||
| Info alerts | ||||
| *Real world example coming soon* | ||||
|  | ||||
| ### Success alert | ||||
| Success alerts | ||||
| *Real world example coming soon* | ||||
|  | ||||
| ### Placement | ||||
|  | ||||
|  | ||||
| ### Accessibility | ||||
| (WAI-ARIA: [https://www.w3.org/TR/wai-aria-practices/#alert](https://www.w3.org/TR/wai-aria-practices/#alert)) | ||||
|  | ||||
| When the component is dynamically displayed, the content is automatically announced by most screen readers. At this time, screen readers do not inform users of alerts that are present when the page loads. | ||||
|  | ||||
| Using color to add meaning only provides a visual indication, which will not be conveyed to users of assistive technologies such as screen readers. Ensure that information denoted by the color is either obvious from the content itself (for example the visible text), or is included through alternative means, such as additional hidden text. | ||||
|  | ||||
| Actions must have a tab index of 0 so that they can be reached by keyboard-only users. | ||||
|  | ||||
| ## Implementation | ||||
|  | ||||
| ### Example Usage | ||||
| **Alert type** | ||||
|  | ||||
| <ha-alert alert-type="error"> | ||||
|   This is an error alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="warning"> | ||||
|   This is an warning alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="info"> | ||||
|   This is an info alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| <ha-alert alert-type="success"> | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="error"> | ||||
|   This is an error alert — check it out! | ||||
| </ha-alert> | ||||
| <ha-alert alert-type="warning"> | ||||
|   This is a warning alert — check it out! | ||||
| </ha-alert> | ||||
| <ha-alert alert-type="info"> | ||||
|   This is an info alert — check it out! | ||||
| </ha-alert> | ||||
| <ha-alert alert-type="success"> | ||||
|   This is a success alert — check it out! | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
| **Title** | ||||
|  | ||||
| The `title ` option should not be used without a description. | ||||
|  | ||||
| <ha-alert alert-type="success" title="Success"> | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="success" title="Success"> | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
| **Dismissable** | ||||
|  | ||||
| <ha-alert alert-type="success" dismissable> | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="success" dismissable> | ||||
|   This is an success alert — check it out! | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
| **Slotted action** | ||||
|  | ||||
| <ha-alert alert-type="success"> | ||||
|   This is an success alert — check it out! | ||||
|   <mwc-button slot="action" label="Undo"></mwc-button> | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="success"> | ||||
|   This is an success alert — check it out! | ||||
|   <mwc-button slot="action" label="Undo"></mwc-button> | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
| **Slotted icon** | ||||
|  | ||||
| *Documentation coming soon* | ||||
|  | ||||
| **Right to left** | ||||
|  | ||||
| <ha-alert alert-type="success" rtl> | ||||
|   This is an info alert — check it out! | ||||
| </ha-alert> | ||||
|  | ||||
| ```html | ||||
| <ha-alert alert-type="success" rtl> | ||||
|   This is an info alert — check it out! | ||||
| </ha-alert> | ||||
| ``` | ||||
|  | ||||
| ### API | ||||
| **Properties/Attributes** | ||||
|  | ||||
| | Name        | Type    | Default | Description                                           | | ||||
| |-------------|---------|---------|-------------------------------------------------------| | ||||
| | title       | string  | ``      | Title to display.                                     | | ||||
| | alertType   | string  | `info`  | Severity level that set a distinctive icon and color. | | ||||
| | dismissable | boolean | `false` | Gives the option to close the alert.                  | | ||||
| | icon        | string  | ``      | Icon to display.                                      | | ||||
| | action      | string  | ``      | Add an action button to the alert.                    | | ||||
| | rtl         | boolean | `false` | Support languages that use right-to-left.             | | ||||
|  | ||||
| **Events** | ||||
|  | ||||
| *Documentation coming soon* | ||||
|  | ||||
| **CSS Custom Properties** | ||||
|  | ||||
| *Documentation coming soon* | ||||
| @@ -1,15 +1,19 @@ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import "@material/mwc-button/mwc-button"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||
| import "../../../../src/components/ha-alert"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-logo-svg"; | ||||
| 
 | ||||
| const alerts: { | ||||
|   title?: string; | ||||
|   description: string | TemplateResult; | ||||
|   type: "info" | "warning" | "error" | "success"; | ||||
|   dismissable?: boolean; | ||||
|   action?: string; | ||||
|   rtl?: boolean; | ||||
|   iconSlot?: TemplateResult; | ||||
|   actionSlot?: TemplateResult; | ||||
| }[] = [ | ||||
|   { | ||||
|     title: "Test info alert", | ||||
| @@ -73,13 +77,35 @@ const alerts: { | ||||
|     title: "Error with action", | ||||
|     description: "This is a test error alert with action", | ||||
|     type: "error", | ||||
|     action: "restart", | ||||
|     actionSlot: html`<mwc-button slot="action" label="restart"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Unsaved data", | ||||
|     description: "You have unsaved data", | ||||
|     type: "warning", | ||||
|     action: "save", | ||||
|     actionSlot: html`<mwc-button slot="action" label="save"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Slotted icon", | ||||
|     description: "Alert with slotted icon", | ||||
|     type: "warning", | ||||
|     iconSlot: html`<span slot="icon" class="image">
 | ||||
|       <ha-logo-svg></ha-logo-svg> | ||||
|     </span>`,
 | ||||
|   }, | ||||
|   { | ||||
|     title: "Slotted image", | ||||
|     description: "Alert with slotted image", | ||||
|     type: "warning", | ||||
|     iconSlot: html`<span slot="icon" class="image"
 | ||||
|       ><img src="https://www.home-assistant.io/images/home-assistant-logo.svg" | ||||
|     /></span>`, | ||||
|   }, | ||||
|   { | ||||
|     title: "Slotted action", | ||||
|     description: "Alert with slotted action", | ||||
|     type: "info", | ||||
|     actionSlot: html`<mwc-button slot="action" label="action"></mwc-button>`, | ||||
|   }, | ||||
|   { | ||||
|     description: "Dismissable information (RTL)", | ||||
| @@ -91,7 +117,7 @@ const alerts: { | ||||
|     title: "Error with action", | ||||
|     description: "This is a test error alert with action (RTL)", | ||||
|     type: "error", | ||||
|     action: "restart", | ||||
|     actionSlot: html`<mwc-button slot="action" label="restart"></mwc-button>`, | ||||
|     rtl: true, | ||||
|   }, | ||||
|   { | ||||
| @@ -102,11 +128,14 @@ const alerts: { | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-ha-alert") | ||||
| @customElement("demo-components-ha-alert") | ||||
| export class DemoHaAlert extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-card header="ha-alert demo"> | ||||
|       ${["light", "dark"].map( | ||||
|         (mode) => html` | ||||
|           <div class=${mode}> | ||||
|             <ha-card header="ha-alert ${mode} demo"> | ||||
|               <div class="card-content"> | ||||
|                 ${alerts.map( | ||||
|                   (alert) => html` | ||||
| @@ -114,22 +143,45 @@ export class DemoHaAlert extends LitElement { | ||||
|                       .title=${alert.title || ""} | ||||
|                       .alertType=${alert.type} | ||||
|                       .dismissable=${alert.dismissable || false} | ||||
|                 .actionText=${alert.action || ""} | ||||
|                       .rtl=${alert.rtl || false} | ||||
|                     > | ||||
|                 ${alert.description} | ||||
|                       ${alert.iconSlot} ${alert.description} ${alert.actionSlot} | ||||
|                     </ha-alert> | ||||
|                   ` | ||||
|                 )} | ||||
|               </div> | ||||
|             </ha-card> | ||||
|           </div> | ||||
|         ` | ||||
|       )} | ||||
|     `;
 | ||||
|   } | ||||
| 
 | ||||
|   firstUpdated(changedProps) { | ||||
|     super.firstUpdated(changedProps); | ||||
|     applyThemesOnElement(this.shadowRoot!.querySelector(".dark"), { | ||||
|       default_theme: "default", | ||||
|       default_dark_theme: "default", | ||||
|       themes: {}, | ||||
|       darkMode: true, | ||||
|       theme: "default", | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       :host { | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         justify-content: space-between; | ||||
|       } | ||||
|       .dark, | ||||
|       .light { | ||||
|         display: block; | ||||
|         background-color: var(--primary-background-color); | ||||
|         padding: 0 50px; | ||||
|       } | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
|       } | ||||
|       ha-alert { | ||||
| @@ -142,8 +194,17 @@ export class DemoHaAlert extends LitElement { | ||||
|         align-items: center; | ||||
|         justify-content: space-between; | ||||
|       } | ||||
|       span { | ||||
|         margin-right: 16px; | ||||
|       .image { | ||||
|         display: inline-flex; | ||||
|         height: 100%; | ||||
|         align-items: center; | ||||
|       } | ||||
|       img { | ||||
|         max-height: 24px; | ||||
|         width: 24px; | ||||
|       } | ||||
|       mwc-button { | ||||
|         --mdc-theme-primary: var(--primary-text-color); | ||||
|       } | ||||
|     `;
 | ||||
|   } | ||||
| @@ -151,6 +212,6 @@ export class DemoHaAlert extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-alert": DemoHaAlert; | ||||
|     "demo-components-ha-alert": DemoHaAlert; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/components/ha-bar.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/components/ha-bar.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Progress Bars | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { classMap } from "lit/directives/class-map"; | ||||
| import "../../../src/components/ha-bar"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-bar"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| 
 | ||||
| const bars: { | ||||
|   min?: number; | ||||
| @@ -34,7 +34,7 @@ const bars: { | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-ha-bar") | ||||
| @customElement("demo-components-ha-bar") | ||||
| export class DemoHaBar extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
| @@ -80,6 +80,6 @@ export class DemoHaBar extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-bar": DemoHaBar; | ||||
|     "demo-components-ha-bar": DemoHaBar; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/components/ha-chips.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/components/ha-chips.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Chips | ||||
| --- | ||||
| @@ -1,9 +1,10 @@ | ||||
| import { mdiHomeAssistant } from "@mdi/js"; | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/ha-chip"; | ||||
| import "../../../src/components/ha-svg-icon"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-chip"; | ||||
| import "../../../../src/components/ha-chip-set"; | ||||
| import "../../../../src/components/ha-svg-icon"; | ||||
| 
 | ||||
| const chips: { | ||||
|   icon?: string; | ||||
| @@ -22,8 +23,8 @@ const chips: { | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-ha-chip") | ||||
| export class DemoHaChip extends LitElement { | ||||
| @customElement("demo-components-ha-chips") | ||||
| export class DemoHaChips extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-card header="ha-chip demo"> | ||||
| @@ -41,6 +42,23 @@ export class DemoHaChip extends LitElement { | ||||
|           )} | ||||
|         </div> | ||||
|       </ha-card> | ||||
|       <ha-card header="ha-chip-set demo"> | ||||
|         <div class="card-content"> | ||||
|           <ha-chip-set> | ||||
|             ${chips.map( | ||||
|               (chip) => html` | ||||
|                 <ha-chip .hasIcon=${chip.icon !== undefined}> | ||||
|                   ${chip.icon | ||||
|                     ? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
 | ||||
|                       </ha-svg-icon>` | ||||
|                     : ""} | ||||
|                   ${chip.content} | ||||
|                 </ha-chip> | ||||
|               ` | ||||
|             )} | ||||
|           </ha-chip-set> | ||||
|         </div> | ||||
|       </ha-card> | ||||
|     `;
 | ||||
|   } | ||||
| 
 | ||||
| @@ -50,12 +68,19 @@ export class DemoHaChip extends LitElement { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
|       } | ||||
|       ha-chip { | ||||
|         margin-bottom: 4px; | ||||
|       } | ||||
|       .card-content { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|       } | ||||
|     `;
 | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-chip": DemoHaChip; | ||||
|     "demo-components-ha-chips": DemoHaChips; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/components/ha-faded.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/components/ha-faded.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Faded Content | ||||
| --- | ||||
							
								
								
									
										88
									
								
								gallery/src/pages/components/ha-faded.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								gallery/src/pages/components/ha-faded.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | ||||
| import { css, html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-faded"; | ||||
| import "../../../../src/components/ha-markdown"; | ||||
|  | ||||
| const LONG_TEXT = ` | ||||
| Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nunc laoreet velit ut elit volutpat, eget ultrices odio lacinia. In imperdiet malesuada est, nec sagittis metus ultricies quis. Sed nisl ex, convallis porttitor ante quis, hendrerit tristique justo. Mauris pharetra venenatis augue, eu maximus sem cursus in. Quisque sed consequat risus. Suspendisse facilisis ligula a odio consectetur condimentum. Curabitur vehicula elit nec augue mollis, et volutpat massa dictum. | ||||
|  | ||||
| Nam pellentesque auctor rutrum. Suspendisse elit est, sodales vel diam nec, porttitor faucibus massa. Ut pretium ac orci eu pharetra. Praesent in nibh at magna viverra rutrum eu vitae tortor. Etiam eget sem ex. Fusce tristique odio nec lacus mattis, vitae tempor nunc malesuada. Maecenas faucibus magna vel libero maximus egestas. Vestibulum luctus semper velit, in lobortis risus tempus non. Curabitur bibendum ornare commodo. Quisque commodo neque sit amet tincidunt lacinia. Proin elementum ante velit, eu congue nulla semper quis. Pellentesque consequat vel nunc at scelerisque. Mauris sit amet venenatis diam, blandit viverra leo. Integer commodo laoreet orci. | ||||
|  | ||||
| Curabitur ipsum tortor, sodales ut augue sed, commodo porttitor libero. Pellentesque molestie vitae mi consectetur tempor. In sed lectus consequat, lobortis neque non, semper ipsum. Etiam eget ex et nibh sagittis pulvinar lacinia ac mauris. Aenean ligula eros, viverra ac nibh at, venenatis semper quam. Sed interdum ligula sit amet massa tincidunt tincidunt. Suspendisse potenti. Aliquam egestas facilisis est, sed faucibus erat scelerisque id. Duis dolor quam, viverra vitae orci euismod, laoreet pellentesque justo. Nunc malesuada non erat at ullamcorper. Mauris eget posuere odio. Vestibulum turpis nunc, pharetra eget ante in, feugiat mollis justo. Proin porttitor, diam nec vulputate pretium, tellus arcu rhoncus turpis, a blandit nisi nulla quis arcu. Nunc ac ullamcorper ligula, nec facilisis leo. | ||||
|  | ||||
| In vitae eros sollicitudin, iaculis ex eget, egestas orci. Etiam sed pretium lorem. Nam nisi enim, consectetur sit amet semper ac, semper pharetra diam. In pulvinar neque sapien, ac ullamcorper est lacinia a. Etiam tincidunt velit sed diam malesuada, eu ornare ex consectetur. Phasellus in imperdiet tellus. Sed bibendum, dui sit amet fringilla aliquet, enim odio sollicitudin lorem, vel semper turpis mauris vel mauris. Aenean congue magna ac massa cursus, in dictum orci commodo. Pellentesque mollis velit in sollicitudin tincidunt. Vestibulum et efficitur nulla. | ||||
|  | ||||
| Quisque posuere, velit sed porttitor dapibus, neque augue fringilla felis, eu luctus nisi nisl nec ipsum. Curabitur pellentesque ac lectus eget ultricies. Vestibulum est dolor, lacinia pharetra vulputate a, facilisis a magna. Nam vitae arcu nibh. Praesent finibus blandit ante, ac gravida ex mollis eget. Donec quam est, pulvinar vitae neque ut, bibendum aliquam erat. Nullam mollis arcu at sem tincidunt, in tristique lectus facilisis. Aenean ut lacus vel nisl finibus iaculis non a turpis. Integer eget ipsum ante. Donec nunc neque, vestibulum ac magna ac, posuere scelerisque dui. Pellentesque massa nibh, rhoncus id dolor quis, placerat posuere turpis. Donec aliquet augue nisi, eu finibus dui auctor et. Vestibulum eu varius lorem. Quisque lectus ante, malesuada pretium risus eget, interdum mattis enim. | ||||
| `; | ||||
|  | ||||
| const SMALL_TEXT = "Lorem ipsum dolor sit amet, consectetur adipiscing elit."; | ||||
|  | ||||
| @customElement("demo-components-ha-faded") | ||||
| export class DemoHaFaded extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-card header="ha-faded demo"> | ||||
|         <div class="card-content"> | ||||
|           <h3>Long text directly as slotted content</h3> | ||||
|           <ha-faded>${LONG_TEXT}</ha-faded> | ||||
|           <h3>Long text with slotted element</h3> | ||||
|           <ha-faded><span>${LONG_TEXT}</span></ha-faded> | ||||
|           <h3>No text</h3> | ||||
|           <ha-faded><span></span></ha-faded> | ||||
|           <h3>Smal text</h3> | ||||
|           <ha-faded><span>${SMALL_TEXT}</span></ha-faded> | ||||
|           <h3>Long text in markdown</h3> | ||||
|           <ha-faded> | ||||
|             <ha-markdown .content=${LONG_TEXT}> </ha-markdown> | ||||
|           </ha-faded> | ||||
|           <h3>Missing 1px from hiding</h3> | ||||
|           <ha-faded faded-height="87"> | ||||
|             <span> | ||||
|               Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nunc | ||||
|               laoreet velit ut elit volutpat, eget ultrices odio lacinia. In | ||||
|               imperdiet malesuada est, nec sagittis metus ultricies quis. Sed | ||||
|               nisl ex, convallis porttitor ante quis, hendrerit tristique justo. | ||||
|               Mauris pharetra venenatis augue, eu maximus sem cursus in. Quisque | ||||
|               sed consequat risus. Suspendisse facilisis ligula a odio | ||||
|               consectetur condimentum. Curabitur vehicula elit nec augue mollis, | ||||
|               et volutpat massa dictum. Nam pellentesque auctor rutrum. | ||||
|               Suspendisse elit est, sodales vel diam nec, porttitor faucibus | ||||
|               massa. Ut pretium ac orci eu pharetra. | ||||
|             </span> | ||||
|           </ha-faded> | ||||
|           <h3>1px over hiding point</h3> | ||||
|           <ha-faded faded-height="85"> | ||||
|             <span> | ||||
|               Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nunc | ||||
|               laoreet velit ut elit volutpat, eget ultrices odio lacinia. In | ||||
|               imperdiet malesuada est, nec sagittis metus ultricies quis. Sed | ||||
|               nisl ex, convallis porttitor ante quis, hendrerit tristique justo. | ||||
|               Mauris pharetra venenatis augue, eu maximus sem cursus in. Quisque | ||||
|               sed consequat risus. Suspendisse facilisis ligula a odio | ||||
|               consectetur condimentum. Curabitur vehicula elit nec augue mollis, | ||||
|               et volutpat massa dictum. Nam pellentesque auctor rutrum. | ||||
|               Suspendisse elit est, sodales vel diam nec, porttitor faucibus | ||||
|               massa. Ut pretium ac orci eu pharetra. | ||||
|             </span> | ||||
|           </ha-faded> | ||||
|         </div> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-components-ha-faded": DemoHaFaded; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/components/ha-form.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/components/ha-form.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Forms | ||||
| --- | ||||
| @@ -1,11 +1,17 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import "@material/mwc-button"; | ||||
| import { LitElement, TemplateResult, html } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import { computeInitialHaFormData } from "../../../src/components/ha-form/compute-initial-ha-form-data"; | ||||
| import type { HaFormSchema } from "../../../src/components/ha-form/types"; | ||||
| import "../../../src/components/ha-form/ha-form"; | ||||
| import "../components/demo-black-white-row"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import { computeInitialHaFormData } from "../../../../src/components/ha-form/compute-initial-ha-form-data"; | ||||
| import type { HaFormSchema } from "../../../../src/components/ha-form/types"; | ||||
| import "../../../../src/components/ha-form/ha-form"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||
| 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"; | ||||
| 
 | ||||
| const SCHEMAS: { | ||||
|   title: string; | ||||
| @@ -14,6 +20,49 @@ const SCHEMAS: { | ||||
|   schema: HaFormSchema[]; | ||||
|   data?: Record<string, any>; | ||||
| }[] = [ | ||||
|   { | ||||
|     title: "Selectors", | ||||
|     translations: { | ||||
|       addon: "Addon", | ||||
|       entity: "Entity", | ||||
|       device: "Device", | ||||
|       area: "Area", | ||||
|       target: "Target", | ||||
|       number: "Number", | ||||
|       boolean: "Boolean", | ||||
|       time: "Time", | ||||
|       action: "Action", | ||||
|       text: "Text", | ||||
|       text_multiline: "Text Multiline", | ||||
|       object: "Object", | ||||
|       select: "Select", | ||||
|     }, | ||||
|     schema: [ | ||||
|       { name: "addon", selector: { addon: {} } }, | ||||
|       { name: "entity", selector: { entity: {} } }, | ||||
|       { | ||||
|         name: "Attribute", | ||||
|         selector: { attribute: { entity_id: "" } }, | ||||
|       }, | ||||
|       { name: "Device", selector: { device: {} } }, | ||||
|       { name: "Duration", selector: { duration: {} } }, | ||||
|       { 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"] }, | ||||
|         }, | ||||
|       }, | ||||
|     ], | ||||
|   }, | ||||
|   { | ||||
|     title: "Authentication", | ||||
|     translations: { | ||||
| @@ -50,13 +99,11 @@ const SCHEMAS: { | ||||
|       { | ||||
|         type: "boolean", | ||||
|         name: "bool", | ||||
|         optional: true, | ||||
|         default: false, | ||||
|       }, | ||||
|       { | ||||
|         type: "integer", | ||||
|         name: "int", | ||||
|         optional: true, | ||||
|         default: 10, | ||||
|       }, | ||||
|       { | ||||
| @@ -67,7 +114,6 @@ const SCHEMAS: { | ||||
|       { | ||||
|         type: "string", | ||||
|         name: "string", | ||||
|         optional: true, | ||||
|         default: "Default", | ||||
|       }, | ||||
|       { | ||||
| @@ -77,7 +123,6 @@ const SCHEMAS: { | ||||
|           ["other", "other"], | ||||
|         ], | ||||
|         name: "select", | ||||
|         optional: true, | ||||
|         default: "default", | ||||
|       }, | ||||
|       { | ||||
| @@ -87,7 +132,6 @@ const SCHEMAS: { | ||||
|           other: "Other", | ||||
|         }, | ||||
|         name: "multi", | ||||
|         optional: true, | ||||
|         default: ["default"], | ||||
|       }, | ||||
|       { | ||||
| @@ -108,7 +152,6 @@ const SCHEMAS: { | ||||
|       { | ||||
|         type: "integer", | ||||
|         name: "int with default", | ||||
|         optional: true, | ||||
|         default: 10, | ||||
|       }, | ||||
|       { | ||||
| @@ -122,7 +165,6 @@ const SCHEMAS: { | ||||
|       { | ||||
|         type: "integer", | ||||
|         name: "int range optional", | ||||
|         optional: true, | ||||
|         valueMin: 0, | ||||
|         valueMax: 10, | ||||
|       }, | ||||
| @@ -148,7 +190,6 @@ const SCHEMAS: { | ||||
|           ["other", "Other"], | ||||
|         ], | ||||
|         name: "select optional", | ||||
|         optional: true, | ||||
|       }, | ||||
|       { | ||||
|         type: "select", | ||||
| @@ -161,7 +202,6 @@ const SCHEMAS: { | ||||
|           ["option", "1000"], | ||||
|         ], | ||||
|         name: "select many otions", | ||||
|         optional: true, | ||||
|         default: "default", | ||||
|       }, | ||||
|     ], | ||||
| @@ -190,7 +230,6 @@ const SCHEMAS: { | ||||
|           option: "1000", | ||||
|         }, | ||||
|         name: "multi many otions", | ||||
|         optional: true, | ||||
|         default: ["default"], | ||||
|       }, | ||||
|     ], | ||||
| @@ -239,23 +278,35 @@ const SCHEMAS: { | ||||
|         valueMin: 1, | ||||
|         valueMax: 65535, | ||||
|         name: "port", | ||||
|         optional: true, | ||||
|         default: 80, | ||||
|       }, | ||||
|       { type: "string", name: "path", optional: true, default: "/" }, | ||||
|       { type: "boolean", name: "ssl", optional: true, default: false }, | ||||
|       { type: "string", name: "path", default: "/" }, | ||||
|       { type: "boolean", name: "ssl", default: false }, | ||||
|     ], | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-ha-form") | ||||
| @customElement("demo-components-ha-form") | ||||
| class DemoHaForm extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
| 
 | ||||
|   private data = SCHEMAS.map( | ||||
|     ({ schema, data }) => data || computeInitialHaFormData(schema) | ||||
|   ); | ||||
| 
 | ||||
|   private disabled = SCHEMAS.map(() => false); | ||||
| 
 | ||||
|   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 { | ||||
|     return html` | ||||
|       ${SCHEMAS.map((info, idx) => { | ||||
| @@ -278,6 +329,7 @@ class DemoHaForm extends LitElement { | ||||
|               (slot) => html` | ||||
|                 <ha-form | ||||
|                   slot=${slot} | ||||
|                   .hass=${this.hass} | ||||
|                   .data=${this.data[idx]} | ||||
|                   .schema=${info.schema} | ||||
|                   .error=${info.error} | ||||
| @@ -301,6 +353,6 @@ class DemoHaForm extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-form": DemoHaForm; | ||||
|     "demo-components-ha-form": DemoHaForm; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/components/ha-label-badge.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/components/ha-label-badge.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Label Badge | ||||
| --- | ||||
| @@ -1,7 +1,7 @@ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../src/components/ha-label-badge"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../../src/components/ha-label-badge"; | ||||
| import "../../../../src/components/ha-card"; | ||||
| 
 | ||||
| const colors = ["#03a9f4", "#ffa600", "#43a047"]; | ||||
| 
 | ||||
| @@ -50,7 +50,7 @@ const badges: { | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-ha-label-badge") | ||||
| @customElement("demo-components-ha-label-badge") | ||||
| export class DemoHaLabelBadge extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
| @@ -117,6 +117,6 @@ export class DemoHaLabelBadge extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-label-badge": DemoHaLabelBadge; | ||||
|     "demo-components-ha-label-badge": DemoHaLabelBadge; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/components/ha-selector.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/components/ha-selector.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Target Selectors | ||||
| --- | ||||
| @@ -2,16 +2,16 @@ | ||||
| import "@material/mwc-button"; | ||||
| import { LitElement, TemplateResult, css, html } from "lit"; | ||||
| import { customElement, state } from "lit/decorators"; | ||||
| import "../../../src/components/ha-selector/ha-selector"; | ||||
| import "../../../src/components/ha-settings-row"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../src/types"; | ||||
| import "../components/demo-black-white-row"; | ||||
| import { BlueprintInput } from "../../../src/data/blueprint"; | ||||
| 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/components/ha-selector/ha-selector"; | ||||
| import "../../../../src/components/ha-settings-row"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import type { HomeAssistant } from "../../../../src/types"; | ||||
| import "../../components/demo-black-white-row"; | ||||
| import { BlueprintInput } from "../../../../src/data/blueprint"; | ||||
| 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"; | ||||
| 
 | ||||
| const SCHEMAS: { | ||||
|   name: string; | ||||
| @@ -21,7 +21,12 @@ const SCHEMAS: { | ||||
|     name: "One of each", | ||||
|     input: { | ||||
|       entity: { name: "Entity", selector: { entity: {} } }, | ||||
|       attribute: { | ||||
|         name: "Attribute", | ||||
|         selector: { attribute: { entity_id: "" } }, | ||||
|       }, | ||||
|       device: { name: "Device", selector: { device: {} } }, | ||||
|       duration: { name: "Duration", selector: { duration: {} } }, | ||||
|       addon: { name: "Addon", selector: { addon: {} } }, | ||||
|       area: { name: "Area", selector: { area: {} } }, | ||||
|       target: { name: "Target", selector: { target: {} } }, | ||||
| @@ -48,10 +53,19 @@ const SCHEMAS: { | ||||
|       boolean: { name: "Boolean", selector: { boolean: {} } }, | ||||
|       time: { name: "Time", selector: { time: {} } }, | ||||
|       action: { name: "Action", selector: { action: {} } }, | ||||
|       text: { name: "Text", selector: { text: { multiline: false } } }, | ||||
|       text: { | ||||
|         name: "Text", | ||||
|         selector: { text: {} }, | ||||
|       }, | ||||
|       password: { | ||||
|         name: "Password", | ||||
|         selector: { text: { type: "password" } }, | ||||
|       }, | ||||
|       text_multiline: { | ||||
|         name: "Text multiline", | ||||
|         selector: { text: { multiline: true } }, | ||||
|         selector: { | ||||
|           text: { multiline: true }, | ||||
|         }, | ||||
|       }, | ||||
|       object: { name: "Object", selector: { object: {} } }, | ||||
|       select: { | ||||
| @@ -62,7 +76,7 @@ const SCHEMAS: { | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-ha-selector") | ||||
| @customElement("demo-components-ha-selector") | ||||
| class DemoHaSelector extends LitElement { | ||||
|   @state() private hass!: HomeAssistant; | ||||
| 
 | ||||
| @@ -126,6 +140,6 @@ class DemoHaSelector extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-ha-selector": DemoHaSelector; | ||||
|     "demo-components-ha-selector": DemoHaSelector; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										24
									
								
								gallery/src/pages/concepts/home.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								gallery/src/pages/concepts/home.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| --- | ||||
| title: Home | ||||
| --- | ||||
|  | ||||
| # Welcome to Home Assistant Design | ||||
|  | ||||
| This portal aims to aid designers and developers on improving the Home Assistant interface. It consists of working code, resources and guidelines. | ||||
|  | ||||
| ## Home Assistant interface | ||||
| The Home Assistant frontend allows users to browse and control the state of their home, manage their automations and configure integrations. The frontend is designed as a mobile-first experience. It is a progressive web application and offers an app-like experience to our users. The Home Assistant frontend needs to be fast. But it also needs to work on a wide range of old devices. | ||||
|  | ||||
| ### Material Design | ||||
| The Home Assistant interface is based on Material Design. It's a design system created by Google to quickly build high-quality digital experiences. Components and guidelines that are custom made for Home Assistant are documented on this portal. For all other components check <a href="https://material.io" rel="noopener noreferrer" target="_blank">material.io</a>. | ||||
|  | ||||
| ## Designers | ||||
| We want to make it as easy for designers to contribute as it is for developers. There’s a lot a designer can contribute to: | ||||
|  | ||||
| - Meet us at <a href="https://discord.gg/BPBc8rZ9" rel="noopener noreferrer" target="_blank">devs_ux Discord</a>. Feel free to share your designs, user test or strategic ideas. | ||||
| - Start designing with our <a href="https://www.figma.com/community/file/967153512097289521/Home-Assistant-DesignKit" rel="noopener noreferrer" target="_blank">Figma DesignKit</a>. | ||||
| - Find the lates UX <a href="https://github.com/home-assistant/frontend/discussions?discussions_q=label%3Aux" rel="noopener noreferrer" target="_blank">discussions</a> and <a href="https://github.com/home-assistant/frontend/labels/ux" rel="noopener noreferrer" target="_blank">issues</a> on GitHub. Everyone can start a new issue or discussion! | ||||
|  | ||||
|  | ||||
| ## Developers | ||||
| Everything you need to get started developing can be found in our <a href="https://developers.home-assistant.io" rel="noopener noreferrer" target="_blank">Home Assistant Developer Docs</a>. | ||||
							
								
								
									
										80
									
								
								gallery/src/pages/design.home-assistant.io/editing.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								gallery/src/pages/design.home-assistant.io/editing.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | ||||
| --- | ||||
| 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. | ||||
|  | ||||
| ## Development | ||||
|  | ||||
| You can develop design.home-assistant.io locally by checking out [the Home Assistant frontend repository](https://github.com/home-assistant/frontend). The command to run the gallery is `gallery/script/develop_gallery`. It will automatically open a browser window and load the development version of the website. | ||||
|  | ||||
| ## Creating a page | ||||
|  | ||||
| Navigate to the [the pages folder][pages-folder] on GitHub. If the folder for your category does not exist yet, create it. Create a new Markdown file inside this folder for your description, ie `usability.markdown`. This filename will be used in the URL. Add the following content: | ||||
|  | ||||
| ```markdown | ||||
| --- | ||||
| title: My new page | ||||
| --- | ||||
|  | ||||
| Hello and welcome to my new page! | ||||
| ``` | ||||
|  | ||||
| Once saved, the page will be automatically added to the bottom of the sidebar. The title specified in the header will be shown as the page title and used in the sidebar. | ||||
|  | ||||
| ## Linking the page in the sidebar | ||||
|  | ||||
| By default the sidebar will gather all pages and group them by category. You can override the order of the categories, define a name for categories and change the order of the pages in [`sidebar.js`](https://github.com/home-assistant/frontend/blob/dev/gallery/sidebar.js). | ||||
|  | ||||
| Any category not listed in `sidebar.js` will be placed at the end of the sidebar. | ||||
|  | ||||
| Any page not listed in `sidebar.js` will be placed at the end of its category. | ||||
|  | ||||
| ## Adding a demo to a page | ||||
|  | ||||
| Create a file next to the description file with the same name as the description file, but with the `.ts` extension: `usability.ts`. For this example, we assume that the category folder that contains `usability.markdown` and `usability.ts` is called `user-experience`. Add the following content to `usability.ts`: | ||||
|  | ||||
| ```ts | ||||
| 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> | ||||
|       </ha-card> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   static get styles() { | ||||
|     return css` | ||||
|       ha-card { | ||||
|         max-width: 600px; | ||||
|         margin: 24px auto; | ||||
|       } | ||||
|     `; | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-user-experience-usability": DemoUserExperienceUsability; | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Note that the demo deosn't need to render anything itself. It can also be used to declare web components to be used by the page description. Because page descriptions are using markdown, they can embed any HTML. | ||||
|  | ||||
| ## Publishing changes | ||||
|  | ||||
| The website is automatically published whenever the source files in the `dev` branch change. So to get your changes published, open a pull request with your changes. | ||||
|  | ||||
| [pages-folder]: https://github.com/home-assistant/frontend/tree/dev/gallery/src/pages | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/alarm-panel-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/alarm-panel-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Alarm Panel Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("alarm_control_panel", "alarm", "disarmed", { | ||||
| @@ -70,7 +70,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-alarm-panel-card") | ||||
| @customElement("demo-lovelace-alarm-panel-card") | ||||
| class DemoAlarmPanelEntity extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -89,6 +89,6 @@ class DemoAlarmPanelEntity extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-alarm-panel-card": DemoAlarmPanelEntity; | ||||
|     "demo-lovelace-alarm-panel-card": DemoAlarmPanelEntity; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/area-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/area-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Area Card | ||||
| --- | ||||
							
								
								
									
										156
									
								
								gallery/src/pages/lovelace/area-card.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								gallery/src/pages/lovelace/area-card.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
|  | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "bed_light", "on", { | ||||
|     friendly_name: "Bed Light", | ||||
|   }), | ||||
|   getEntity("switch", "bed_ac", "on", { | ||||
|     friendly_name: "Ecobee", | ||||
|   }), | ||||
|   getEntity("sensor", "bed_temp", "72", { | ||||
|     friendly_name: "Bedroom Temp", | ||||
|     device_class: "temperature", | ||||
|     unit_of_measurement: "°F", | ||||
|   }), | ||||
|   getEntity("light", "living_room_light", "off", { | ||||
|     friendly_name: "Living Room Light", | ||||
|   }), | ||||
|   getEntity("fan", "living_room", "on", { | ||||
|     friendly_name: "Living Room Fan", | ||||
|   }), | ||||
|   getEntity("sensor", "office_humidity", "73", { | ||||
|     friendly_name: "Office Humidity", | ||||
|     device_class: "humidity", | ||||
|     unit_of_measurement: "%", | ||||
|   }), | ||||
|   getEntity("light", "office", "on", { | ||||
|     friendly_name: "Office Light", | ||||
|   }), | ||||
|   getEntity("fan", "kitchen", "on", { | ||||
|     friendly_name: "Second Office Fan", | ||||
|   }), | ||||
|   getEntity("binary_sensor", "kitchen_door", "on", { | ||||
|     friendly_name: "Office Door", | ||||
|     device_class: "door", | ||||
|   }), | ||||
| ]; | ||||
|  | ||||
| // TODO: Update image here | ||||
| const CONFIGS = [ | ||||
|   { | ||||
|     heading: "Bedroom", | ||||
|     config: ` | ||||
| - type: area | ||||
|   area: bedroom | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Living Room", | ||||
|     config: ` | ||||
| - type: area | ||||
|   area: living_room | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Office", | ||||
|     config: ` | ||||
| - type: area | ||||
|   area: office | ||||
|     `, | ||||
|   }, | ||||
|   { | ||||
|     heading: "Kitchen", | ||||
|     config: ` | ||||
| - type: area | ||||
|   area: kitchen | ||||
|     `, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-lovelace-area-card") | ||||
| class DemoArea extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`; | ||||
|   } | ||||
|  | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     const hass = provideHass(this._demoRoot); | ||||
|     hass.updateTranslations(null, "en"); | ||||
|     hass.updateTranslations("lovelace", "en"); | ||||
|     hass.addEntities(ENTITIES); | ||||
|     hass.mockWS("config/area_registry/list", () => [ | ||||
|       { | ||||
|         name: "Bedroom", | ||||
|         area_id: "bedroom", | ||||
|         picture: "/images/bed.png", | ||||
|       }, | ||||
|       { | ||||
|         name: "Living Room", | ||||
|         area_id: "living_room", | ||||
|         picture: "/images/living_room.png", | ||||
|       }, | ||||
|       { | ||||
|         name: "Office", | ||||
|         area_id: "office", | ||||
|         picture: "/images/office.jpg", | ||||
|       }, | ||||
|       { | ||||
|         name: "Second Office", | ||||
|         area_id: "kitchen", | ||||
|         picture: "/images/kitchen.png", | ||||
|       }, | ||||
|     ]); | ||||
|     hass.mockWS("config/device_registry/list", () => []); | ||||
|     hass.mockWS("config/entity_registry/list", () => [ | ||||
|       { | ||||
|         area_id: "bedroom", | ||||
|         entity_id: "light.bed_light", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "bedroom", | ||||
|         entity_id: "switch.bed_ac", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "bedroom", | ||||
|         entity_id: "sensor.bed_temp", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "living_room", | ||||
|         entity_id: "light.living_room_light", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "living_room", | ||||
|         entity_id: "fan.living_room", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "office", | ||||
|         entity_id: "light.office", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "office", | ||||
|         entity_id: "sensor.office_humidity", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "kitchen", | ||||
|         entity_id: "fan.kitchen", | ||||
|       }, | ||||
|       { | ||||
|         area_id: "kitchen", | ||||
|         entity_id: "binary_sensor.kitchen_door", | ||||
|       }, | ||||
|     ]); | ||||
|   } | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-lovelace-area-card": DemoArea; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/conditional-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/conditional-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Conditional Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "controller_1", "on", { | ||||
| @@ -52,7 +52,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-conditional-card") | ||||
| @customElement("demo-lovelace-conditional-card") | ||||
| class DemoConditional extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -71,6 +71,6 @@ class DemoConditional extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-conditional-card": DemoConditional; | ||||
|     "demo-lovelace-conditional-card": DemoConditional; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/entities-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/entities-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Entities Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "bed_light", "on", { | ||||
| @@ -11,10 +11,10 @@ const ENTITIES = [ | ||||
|   getEntity("group", "kitchen", "on", { | ||||
|     entity_id: ["light.bed_light"], | ||||
|     order: 8, | ||||
|     friendly_name: "Kitchen", | ||||
|     friendly_name: "Kitchen Group", | ||||
|   }), | ||||
|   getEntity("lock", "kitchen_door", "locked", { | ||||
|     friendly_name: "Kitchen Door", | ||||
|     friendly_name: "Kitchen Lock", | ||||
|   }), | ||||
|   getEntity("cover", "kitchen_window", "open", { | ||||
|     friendly_name: "Kitchen Window", | ||||
| @@ -22,7 +22,7 @@ const ENTITIES = [ | ||||
|   }), | ||||
|   getEntity("scene", "romantic_lights", "scening", { | ||||
|     entity_id: ["light.bed_light", "light.ceiling_lights"], | ||||
|     friendly_name: "Romantic lights", | ||||
|     friendly_name: "Romantic Scene", | ||||
|   }), | ||||
|   getEntity("device_tracker", "demo_paulus", "home", { | ||||
|     source_type: "gps", | ||||
| @@ -50,15 +50,51 @@ const ENTITIES = [ | ||||
|     friendly_name: "Ecobee", | ||||
|     supported_features: 1014, | ||||
|   }), | ||||
|   getEntity("input_number", "noise_allowance", 5, { | ||||
|   getEntity("input_number", "number", 5, { | ||||
|     min: 0, | ||||
|     max: 10, | ||||
|     step: 1, | ||||
|     mode: "slider", | ||||
|     unit_of_measurement: "dB", | ||||
|     friendly_name: "Allowed Noise", | ||||
|     friendly_name: "Number", | ||||
|     icon: "mdi:bell-ring", | ||||
|   }), | ||||
|   getEntity("input_boolean", "toggle", "on", { | ||||
|     friendly_name: "Toggle", | ||||
|   }), | ||||
|   getEntity("input_datetime", "date_and_time", "2022-01-10 00:00:00", { | ||||
|     has_date: true, | ||||
|     has_time: true, | ||||
|     editable: true, | ||||
|     year: 2022, | ||||
|     month: 1, | ||||
|     day: 10, | ||||
|     hour: 0, | ||||
|     minute: 0, | ||||
|     second: 0, | ||||
|     timestamp: 1641801600, | ||||
|     friendly_name: "Date and Time", | ||||
|   }), | ||||
|   getEntity("input_select", "dropdown", "Soda", { | ||||
|     friendly_name: "Dropdown", | ||||
|     options: ["Soda", "Beer", "Wine"], | ||||
|   }), | ||||
|   getEntity("input_text", "text", "Inspiration", { | ||||
|     friendly_name: "Text", | ||||
|     mode: "text", | ||||
|   }), | ||||
|   getEntity("timer", "timer", "idle", { | ||||
|     friendly_name: "Timer", | ||||
|     duration: "0:05:00", | ||||
|   }), | ||||
|   getEntity("counter", "counter", "3", { | ||||
|     friendly_name: "Counter", | ||||
|     initial: 0, | ||||
|     step: 1, | ||||
|     minimum: 0, | ||||
|     maximum: 10, | ||||
|   }), | ||||
| 
 | ||||
|   getEntity("light", "unavailable", "unavailable", { | ||||
|     friendly_name: "Bed Light", | ||||
|   }), | ||||
| @@ -70,7 +106,7 @@ const ENTITIES = [ | ||||
|     supported_features: 11, | ||||
|   }), | ||||
|   getEntity("scene", "unavailable", "unavailable", { | ||||
|     friendly_name: "Romantic lights", | ||||
|     friendly_name: "Romantic Scene", | ||||
|   }), | ||||
|   getEntity("device_tracker", "unavailable", "unavailable", { | ||||
|     friendly_name: "Paulus", | ||||
| @@ -105,7 +141,22 @@ const CONFIGS = [ | ||||
|     - light.bed_light | ||||
|     - light.non_existing | ||||
|     - climate.ecobee | ||||
|     - input_number.noise_allowance | ||||
|     - input_number.number | ||||
|     `,
 | ||||
|   }, | ||||
|   { | ||||
|     heading: "Helpers", | ||||
|     config: ` | ||||
| - type: entities | ||||
|   title: Helpers | ||||
|   entities: | ||||
|     - entity: input_boolean.toggle | ||||
|     - entity: input_datetime.date_and_time | ||||
|     - entity: input_number.number | ||||
|     - entity: input_select.dropdown | ||||
|     - entity: input_text.text | ||||
|     - entity: timer.timer | ||||
|     - entity: counter.counter | ||||
|     `,
 | ||||
|   }, | ||||
|   { | ||||
| @@ -120,7 +171,7 @@ const CONFIGS = [ | ||||
|     - lock.kitchen_door | ||||
|     - light.bed_light | ||||
|     - climate.ecobee | ||||
|     - input_number.noise_allowance | ||||
|     - input_number.number | ||||
|   title: Random group | ||||
|     `,
 | ||||
|   }, | ||||
| @@ -136,7 +187,7 @@ const CONFIGS = [ | ||||
|     - lock.kitchen_door | ||||
|     - light.bed_light | ||||
|     - climate.ecobee | ||||
|     - input_number.noise_allowance | ||||
|     - input_number.number | ||||
|   title: Random group | ||||
|   show_header_toggle: false | ||||
|     `,
 | ||||
| @@ -183,7 +234,7 @@ const CONFIGS = [ | ||||
|       icon: mdi:alarm-light | ||||
|       name: Bed Light Custom Icon | ||||
|     - climate.ecobee | ||||
|     - input_number.noise_allowance | ||||
|     - input_number.number | ||||
|   title: Random group | ||||
|   show_header_toggle: false | ||||
|     `,
 | ||||
| @@ -216,7 +267,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-entities-card") | ||||
| @customElement("demo-lovelace-entities-card") | ||||
| class DemoEntities extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -235,6 +286,6 @@ class DemoEntities extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-entities-card": DemoEntities; | ||||
|     "demo-lovelace-entities-card": DemoEntities; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/entity-button-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/entity-button-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Entity Button Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "bed_light", "on", { | ||||
| @@ -68,7 +68,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-entity-button-card") | ||||
| @customElement("demo-lovelace-entity-button-card") | ||||
| class DemoButtonEntity extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -87,6 +87,6 @@ class DemoButtonEntity extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-entity-button-card": DemoButtonEntity; | ||||
|     "demo-lovelace-entity-button-card": DemoButtonEntity; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/entity-filter-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/entity-filter-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Entity Filter Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("device_tracker", "demo_paulus", "work", { | ||||
| @@ -109,7 +109,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-entity-filter-card") | ||||
| @customElement("demo-lovelace-entity-filter-card") | ||||
| class DemoEntityFilter extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -128,6 +128,6 @@ class DemoEntityFilter extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-entity-filter-card": DemoEntityFilter; | ||||
|     "demo-lovelace-entity-filter-card": DemoEntityFilter; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/gauge-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/gauge-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Gauge Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("sensor", "brightness", "12", {}), | ||||
| @@ -106,7 +106,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-gauge-card") | ||||
| @customElement("demo-lovelace-gauge-card") | ||||
| class DemoGaugeEntity extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -125,6 +125,6 @@ class DemoGaugeEntity extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-gauge-card": DemoGaugeEntity; | ||||
|     "demo-lovelace-gauge-card": DemoGaugeEntity; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/glance-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/glance-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Glance Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("device_tracker", "demo_paulus", "home", { | ||||
| @@ -209,7 +209,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-glance-card") | ||||
| @customElement("demo-lovelace-glance-card") | ||||
| class DemoGlanceEntity extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -228,6 +228,6 @@ class DemoGlanceEntity extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-glance-card": DemoGlanceEntity; | ||||
|     "demo-lovelace-glance-card": DemoGlanceEntity; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/grid-and-stack-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/grid-and-stack-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Grid And Stack Card | ||||
| --- | ||||
| @@ -1,9 +1,9 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { mockHistory } from "../../../demo/src/stubs/history"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { mockHistory } from "../../../../demo/src/stubs/history"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "kitchen_lights", "on", { | ||||
| @@ -199,7 +199,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-grid-and-stack-card") | ||||
| @customElement("demo-lovelace-grid-and-stack-card") | ||||
| class DemoStack extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -219,6 +219,6 @@ class DemoStack extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-grid-and-stack-card": DemoStack; | ||||
|     "demo-lovelace-grid-and-stack-card": DemoStack; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/iframe-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/iframe-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Website Card | ||||
| --- | ||||
| @@ -1,6 +1,7 @@ | ||||
| import { html, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../components/demo-cards"; | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const CONFIGS = [ | ||||
|   { | ||||
| @@ -36,15 +37,22 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-iframe-card") | ||||
| @customElement("demo-lovelace-iframe-card") | ||||
| class DemoIframe extends LitElement { | ||||
|   @query("demo-cards") private _demos!: HTMLElement; | ||||
| 
 | ||||
|   protected render(): TemplateResult { | ||||
|     return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`; | ||||
|   } | ||||
| 
 | ||||
|   protected firstUpdated(changedProperties: PropertyValues) { | ||||
|     super.firstUpdated(changedProperties); | ||||
|     provideHass(this._demos); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-iframe-card": DemoIframe; | ||||
|     "demo-lovelace-iframe-card": DemoIframe; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										11
									
								
								gallery/src/pages/lovelace/introduction.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								gallery/src/pages/lovelace/introduction.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| --- | ||||
| title: Introduction | ||||
| --- | ||||
| Lovelace has many different cards. Each card allows the user to tell | ||||
| a different story about what is going on in their house. These cards | ||||
| are very customizable, as no household is the same. | ||||
|  | ||||
| This gallery helps our developers and designers to see all the | ||||
| different states that each card can be in. | ||||
|  | ||||
| Check [the Lovelace documentation](https://www.home-assistant.io/lovelace) for instructions on how to get started with Lovelace. | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/light-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/light-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Light Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "bed_light", "on", { | ||||
| @@ -62,7 +62,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-light-card") | ||||
| @customElement("demo-lovelace-light-card") | ||||
| class DemoLightEntity extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -81,6 +81,6 @@ class DemoLightEntity extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-light-card": DemoLightEntity; | ||||
|     "demo-lovelace-light-card": DemoLightEntity; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/map-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/map-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Map Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("device_tracker", "demo_paulus", "not_home", { | ||||
| @@ -160,7 +160,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-map-card") | ||||
| @customElement("demo-lovelace-map-card") | ||||
| class DemoMap extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -179,6 +179,6 @@ class DemoMap extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-map-card": DemoMap; | ||||
|     "demo-lovelace-map-card": DemoMap; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/markdown-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/markdown-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Markdown Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { mockTemplate } from "../../../demo/src/stubs/template"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { mockTemplate } from "../../../../demo/src/stubs/template"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const CONFIGS = [ | ||||
|   { | ||||
| @@ -253,7 +253,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-markdown-card") | ||||
| @customElement("demo-lovelace-markdown-card") | ||||
| class DemoMarkdown extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -272,6 +272,6 @@ class DemoMarkdown extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-markdown-card": DemoMarkdown; | ||||
|     "demo-lovelace-markdown-card": DemoMarkdown; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/media-control-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/media-control-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Media Control Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { createMediaPlayerEntities } from "../data/media_players"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| import { createMediaPlayerEntities } from "../../data/media_players"; | ||||
| 
 | ||||
| const CONFIGS = [ | ||||
|   { | ||||
| @@ -157,7 +157,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-media-control-card") | ||||
| @customElement("demo-lovelace-media-control-card") | ||||
| class DemoHuiMediaControlCard extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -176,6 +176,6 @@ class DemoHuiMediaControlCard extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-media-control-card": DemoHuiMediaControlCard; | ||||
|     "demo-lovelace-media-control-card": DemoHuiMediaControlCard; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/media-player-row.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/media-player-row.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Media Player Row | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { createMediaPlayerEntities } from "../data/media_players"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| import { createMediaPlayerEntities } from "../../data/media_players"; | ||||
| 
 | ||||
| const CONFIGS = [ | ||||
|   { | ||||
| @@ -54,7 +54,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-media-player-row") | ||||
| @customElement("demo-lovelace-media-player-row") | ||||
| class DemoHuiMediaPlayerRow extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -73,6 +73,6 @@ class DemoHuiMediaPlayerRow extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-media-player-row": DemoHuiMediaPlayerRow; | ||||
|     "demo-lovelace-media-player-rows": DemoHuiMediaPlayerRow; | ||||
|   } | ||||
| } | ||||
| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Picture Elements Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "bed_light", "on", { | ||||
| @@ -124,7 +124,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-picture-elements-card") | ||||
| @customElement("demo-lovelace-picture-elements-card") | ||||
| class DemoPictureElements extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -143,6 +143,6 @@ class DemoPictureElements extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-picture-elements-card": DemoPictureElements; | ||||
|     "demo-lovelace-picture-elements-card": DemoPictureElements; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										3
									
								
								gallery/src/pages/lovelace/picture-entity-card.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/lovelace/picture-entity-card.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| --- | ||||
| title: Picture Entity Card | ||||
| --- | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||
| import { customElement, query } from "lit/decorators"; | ||||
| import { getEntity } from "../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import "../components/demo-cards"; | ||||
| import { getEntity } from "../../../../src/fake_data/entity"; | ||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||
| import "../../components/demo-cards"; | ||||
| 
 | ||||
| const ENTITIES = [ | ||||
|   getEntity("light", "kitchen_lights", "on", { | ||||
| @@ -79,7 +79,7 @@ const CONFIGS = [ | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| @customElement("demo-hui-picture-entity-card") | ||||
| @customElement("demo-lovelace-picture-entity-card") | ||||
| class DemoPictureEntity extends LitElement { | ||||
|   @query("#demos") private _demoRoot!: HTMLElement; | ||||
| 
 | ||||
| @@ -98,6 +98,6 @@ class DemoPictureEntity extends LitElement { | ||||
| 
 | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     "demo-hui-picture-entity-card": DemoPictureEntity; | ||||
|     "demo-lovelace-picture-entity-card": DemoPictureEntity; | ||||
|   } | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user