mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-11-04 00:19:47 +00:00 
			
		
		
		
	Compare commits
	
		
			1099 Commits
		
	
	
		
			template-e
			...
			checkbox-s
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					0bfeb22209 | ||
| 
						 | 
					0e3eed0563 | ||
| 
						 | 
					1b1676cecc | ||
| 
						 | 
					d911fe6a0b | ||
| 
						 | 
					22253a3385 | ||
| 
						 | 
					38640c99e3 | ||
| 
						 | 
					d6df8bddea | ||
| 
						 | 
					ddfc4bd98e | ||
| 
						 | 
					3d6674325c | ||
| 
						 | 
					194829f5b1 | ||
| 
						 | 
					11a77253f4 | ||
| 
						 | 
					67be2343f8 | ||
| 
						 | 
					e9b1b3d853 | ||
| 
						 | 
					8a33d174d7 | ||
| 
						 | 
					226d6216b7 | ||
| 
						 | 
					1925bb01be | ||
| 
						 | 
					82a4806e01 | ||
| 
						 | 
					ce419fae7b | ||
| 
						 | 
					c68b76e2da | ||
| 
						 | 
					342020b420 | ||
| 
						 | 
					1e6e99e3c7 | ||
| 
						 | 
					2e9aafc377 | ||
| 
						 | 
					299c863f49 | ||
| 
						 | 
					c2792a28ba | ||
| 
						 | 
					635a027a8e | ||
| 
						 | 
					a45b8ca8e7 | ||
| 
						 | 
					1e6e945a07 | ||
| 
						 | 
					f71157c24d | ||
| 
						 | 
					e87a2b36cf | ||
| 
						 | 
					5418474f64 | ||
| 
						 | 
					8836ba6ceb | ||
| 
						 | 
					509c5b497a | ||
| 
						 | 
					e00bcc9f48 | ||
| 
						 | 
					bdef9fd040 | ||
| 
						 | 
					c956491ec5 | ||
| 
						 | 
					68bc549d6a | ||
| 
						 | 
					9c64eafc21 | ||
| 
						 | 
					b05e86d442 | ||
| 
						 | 
					fe5f9576c6 | ||
| 
						 | 
					1b282b65b7 | ||
| 
						 | 
					e49664bad3 | ||
| 
						 | 
					2a30b55a43 | ||
| 
						 | 
					9d0b20adce | ||
| 
						 | 
					acd5e1c081 | ||
| 
						 | 
					cc1c5e45b2 | ||
| 
						 | 
					038199c447 | ||
| 
						 | 
					8a1eab7ceb | ||
| 
						 | 
					bc5bd35448 | ||
| 
						 | 
					1795fd56b7 | ||
| 
						 | 
					4a7c33edad | ||
| 
						 | 
					797f60d725 | ||
| 
						 | 
					2427d68aa1 | ||
| 
						 | 
					00c6b0f8ed | ||
| 
						 | 
					7b8d4ab3d6 | ||
| 
						 | 
					07a1a805f6 | ||
| 
						 | 
					d8bab6aba9 | ||
| 
						 | 
					a930e2dc75 | ||
| 
						 | 
					2eb35668fa | ||
| 
						 | 
					07f4e5ac5c | ||
| 
						 | 
					db82a90414 | ||
| 
						 | 
					51a693badf | ||
| 
						 | 
					2aa8f5b352 | ||
| 
						 | 
					93b3b8f985 | ||
| 
						 | 
					92c8bd80b5 | ||
| 
						 | 
					528af0157d | ||
| 
						 | 
					10a77b6278 | ||
| 
						 | 
					03bbf6a582 | ||
| 
						 | 
					63fcb649d2 | ||
| 
						 | 
					4f60a92b92 | ||
| 
						 | 
					0419c1a41f | ||
| 
						 | 
					2d5ae78521 | ||
| 
						 | 
					959134df02 | ||
| 
						 | 
					a9f9fc4ce2 | ||
| 
						 | 
					cfb370a3c8 | ||
| 
						 | 
					353435c8d5 | ||
| 
						 | 
					c8c85d096b | ||
| 
						 | 
					19c9c8f227 | ||
| 
						 | 
					6ea2a29eea | ||
| 
						 | 
					59f3f819a6 | ||
| 
						 | 
					93e8f52880 | ||
| 
						 | 
					02810efcc4 | ||
| 
						 | 
					4b9be7ce16 | ||
| 
						 | 
					f3ec09e480 | ||
| 
						 | 
					8291a84e3e | ||
| 
						 | 
					b0e1f0f73a | ||
| 
						 | 
					a66b966e7d | ||
| 
						 | 
					5f56040c64 | ||
| 
						 | 
					eaccd22267 | ||
| 
						 | 
					27845a7345 | ||
| 
						 | 
					f7ef8180e4 | ||
| 
						 | 
					5958eb9a55 | ||
| 
						 | 
					3ef2912b60 | ||
| 
						 | 
					fa9c6a765a | ||
| 
						 | 
					c4a8899780 | ||
| 
						 | 
					3cc4628d03 | ||
| 
						 | 
					b6c5223221 | ||
| 
						 | 
					cbd6d4251c | ||
| 
						 | 
					fdcbb5b432 | ||
| 
						 | 
					de09e31815 | ||
| 
						 | 
					f55e911313 | ||
| 
						 | 
					465a91dbf3 | ||
| 
						 | 
					835a7833ae | ||
| 
						 | 
					179717d40c | ||
| 
						 | 
					3d4d789f7f | ||
| 
						 | 
					d97fb19f05 | ||
| 
						 | 
					0dd3757df2 | ||
| 
						 | 
					c32a4546f3 | ||
| 
						 | 
					1bb025ccd0 | ||
| 
						 | 
					2b8033a97f | ||
| 
						 | 
					21a3a8c594 | ||
| 
						 | 
					1026e90296 | ||
| 
						 | 
					0eca602e61 | ||
| 
						 | 
					7f75ca81f1 | ||
| 
						 | 
					8af05e2726 | ||
| 
						 | 
					0a478ee1da | ||
| 
						 | 
					a4bdc5a05f | ||
| 
						 | 
					d425767dae | ||
| 
						 | 
					c78382c119 | ||
| 
						 | 
					ee15ddfbc3 | ||
| 
						 | 
					0af14eb77e | ||
| 
						 | 
					583cc4bc8a | ||
| 
						 | 
					2ee92f48e6 | ||
| 
						 | 
					d05e02ab3e | ||
| 
						 | 
					abb9f8e233 | ||
| 
						 | 
					f873ef9b59 | ||
| 
						 | 
					1255b56522 | ||
| 
						 | 
					fd9bb4d8cc | ||
| 
						 | 
					9328576b55 | ||
| 
						 | 
					70a1edd1dd | ||
| 
						 | 
					87e4c209f4 | ||
| 
						 | 
					3d0a5642cc | ||
| 
						 | 
					e211d812ad | ||
| 
						 | 
					0dcf673b87 | ||
| 
						 | 
					cb14e1f20c | ||
| 
						 | 
					52087c0e30 | ||
| 
						 | 
					1b9286db76 | ||
| 
						 | 
					bc92c0b052 | ||
| 
						 | 
					245bb639f2 | ||
| 
						 | 
					8d81ed58c8 | ||
| 
						 | 
					7890ca85a8 | ||
| 
						 | 
					07bab7b264 | ||
| 
						 | 
					5730c14dc1 | ||
| 
						 | 
					f8e8b5ad18 | ||
| 
						 | 
					fd2728c02c | ||
| 
						 | 
					7e2bf920e1 | ||
| 
						 | 
					1f65328f2d | ||
| 
						 | 
					4f731baa00 | ||
| 
						 | 
					5abb3dd8c1 | ||
| 
						 | 
					0a672c55c5 | ||
| 
						 | 
					a6b2299c74 | ||
| 
						 | 
					37cc6709d4 | ||
| 
						 | 
					f4ffbe67e2 | ||
| 
						 | 
					9f32d72a41 | ||
| 
						 | 
					64a117d8ac | ||
| 
						 | 
					ebf0bdc840 | ||
| 
						 | 
					cc0a120bf6 | ||
| 
						 | 
					fe2fe7468f | ||
| 
						 | 
					b12a10ccb5 | ||
| 
						 | 
					2ad2a4b198 | ||
| 
						 | 
					6a62f05657 | ||
| 
						 | 
					4910f60ec4 | ||
| 
						 | 
					d35168e88f | ||
| 
						 | 
					01b3d2aca9 | ||
| 
						 | 
					29e8d1cff0 | ||
| 
						 | 
					4e1d10cc08 | ||
| 
						 | 
					3575d94ca1 | ||
| 
						 | 
					d91546b532 | ||
| 
						 | 
					9f554f4917 | ||
| 
						 | 
					d4720a9244 | ||
| 
						 | 
					5c466712db | ||
| 
						 | 
					6dc7e852ae | ||
| 
						 | 
					785f614bd9 | ||
| 
						 | 
					0a8e27249d | ||
| 
						 | 
					15ee87ee67 | ||
| 
						 | 
					12612a16df | ||
| 
						 | 
					4f449e2600 | ||
| 
						 | 
					7f49f039fd | ||
| 
						 | 
					88dc65bc4e | ||
| 
						 | 
					6edebe18ad | ||
| 
						 | 
					38b3a9205d | ||
| 
						 | 
					4b796b4929 | ||
| 
						 | 
					83cabcac28 | ||
| 
						 | 
					d308c5d9b9 | ||
| 
						 | 
					9f032a61a9 | ||
| 
						 | 
					0f58214ba1 | ||
| 
						 | 
					c48a60cce6 | ||
| 
						 | 
					cd3ffceeff | ||
| 
						 | 
					9be4a00169 | ||
| 
						 | 
					a9c7a39a47 | ||
| 
						 | 
					abcdd60a21 | ||
| 
						 | 
					a94f85a100 | ||
| 
						 | 
					9755bf723f | ||
| 
						 | 
					a71ebcf47e | ||
| 
						 | 
					72695631cd | ||
| 
						 | 
					2af211b543 | ||
| 
						 | 
					6e5e2625d6 | ||
| 
						 | 
					23c1c2f5eb | ||
| 
						 | 
					da85ee5d01 | ||
| 
						 | 
					d408e8653c | ||
| 
						 | 
					cc76ccc3c9 | ||
| 
						 | 
					105a00d3e4 | ||
| 
						 | 
					2c08cba8cc | ||
| 
						 | 
					344b11a204 | ||
| 
						 | 
					1ff5bf0fd5 | ||
| 
						 | 
					c29cf7f77c | ||
| 
						 | 
					193cb46d60 | ||
| 
						 | 
					9dc864d486 | ||
| 
						 | 
					cee166839a | ||
| 
						 | 
					1a60a3c728 | ||
| 
						 | 
					5d946778cb | ||
| 
						 | 
					ac5f85820f | ||
| 
						 | 
					716e100a28 | ||
| 
						 | 
					7b8cb16c12 | ||
| 
						 | 
					00d46424a3 | ||
| 
						 | 
					2a5f940744 | ||
| 
						 | 
					13cc016b36 | ||
| 
						 | 
					a8d49c27c8 | ||
| 
						 | 
					a8522e91b5 | ||
| 
						 | 
					5754f4463d | ||
| 
						 | 
					d4118ade0f | ||
| 
						 | 
					6d80f15a98 | ||
| 
						 | 
					f8aa472409 | ||
| 
						 | 
					df22fd00ca | ||
| 
						 | 
					ce2743a982 | ||
| 
						 | 
					92b32458ad | ||
| 
						 | 
					d57e8a45d3 | ||
| 
						 | 
					551d3ffdf3 | ||
| 
						 | 
					7add6eb736 | ||
| 
						 | 
					a28616d535 | ||
| 
						 | 
					a288fd370f | ||
| 
						 | 
					acd335e249 | ||
| 
						 | 
					da0bfa1945 | ||
| 
						 | 
					3c61d709b5 | ||
| 
						 | 
					ffc92a7b63 | ||
| 
						 | 
					af0c7b5a50 | ||
| 
						 | 
					1904c4d057 | ||
| 
						 | 
					542f169b36 | ||
| 
						 | 
					65a30bf60c | ||
| 
						 | 
					2e51da32f0 | ||
| 
						 | 
					0562242043 | ||
| 
						 | 
					debcdefc21 | ||
| 
						 | 
					0de3f3a332 | ||
| 
						 | 
					4fcb4d449e | ||
| 
						 | 
					408fe25abd | ||
| 
						 | 
					236e5e0b25 | ||
| 
						 | 
					ebe0caba83 | ||
| 
						 | 
					9d33c0cfaf | ||
| 
						 | 
					7962130a0c | ||
| 
						 | 
					9690434cac | ||
| 
						 | 
					7304544c37 | ||
| 
						 | 
					5a3408c242 | ||
| 
						 | 
					16996f25af | ||
| 
						 | 
					0c12586019 | ||
| 
						 | 
					93a1adaa56 | ||
| 
						 | 
					83e65e2cc6 | ||
| 
						 | 
					36586b798e | ||
| 
						 | 
					20c351949f | ||
| 
						 | 
					b63bd92d81 | ||
| 
						 | 
					9f3bb7f4d6 | ||
| 
						 | 
					73bb346c00 | ||
| 
						 | 
					33703a3b53 | ||
| 
						 | 
					b7a4f97eca | ||
| 
						 | 
					dd4efe0f51 | ||
| 
						 | 
					7e0522c3b3 | ||
| 
						 | 
					e682abfb75 | ||
| 
						 | 
					24e202a3d7 | ||
| 
						 | 
					ac9a881ab5 | ||
| 
						 | 
					4d287a1f83 | ||
| 
						 | 
					b8d6b1ebdd | ||
| 
						 | 
					8ca1b9320d | ||
| 
						 | 
					cba3992d2b | ||
| 
						 | 
					96d6e337be | ||
| 
						 | 
					959f7ae046 | ||
| 
						 | 
					9572a58764 | ||
| 
						 | 
					393ae9e5dc | ||
| 
						 | 
					63e10314bd | ||
| 
						 | 
					b599417a37 | ||
| 
						 | 
					899eab4e5c | ||
| 
						 | 
					3f21c87a3d | ||
| 
						 | 
					c296a60bab | ||
| 
						 | 
					5f78f18cb4 | ||
| 
						 | 
					0b8d356865 | ||
| 
						 | 
					e8d1318a5b | ||
| 
						 | 
					07ce07c4a5 | ||
| 
						 | 
					a07220f383 | ||
| 
						 | 
					f21ed24a49 | ||
| 
						 | 
					e3c38b93f4 | ||
| 
						 | 
					b398727413 | ||
| 
						 | 
					9bc2ab29a1 | ||
| 
						 | 
					51f1ff26f1 | ||
| 
						 | 
					97d5e6512d | ||
| 
						 | 
					b76c67fc9b | ||
| 
						 | 
					b96a70cd55 | ||
| 
						 | 
					982ab93cdb | ||
| 
						 | 
					c7f4e1152d | ||
| 
						 | 
					519988326b | ||
| 
						 | 
					b518f4b03c | ||
| 
						 | 
					5493fdfcb7 | ||
| 
						 | 
					179767e9f8 | ||
| 
						 | 
					25b3bb1285 | ||
| 
						 | 
					841c8ab1f1 | ||
| 
						 | 
					1ce17e2847 | ||
| 
						 | 
					a09b206b0e | ||
| 
						 | 
					bb4617c53b | ||
| 
						 | 
					cfd18bfb74 | ||
| 
						 | 
					e225d6f546 | ||
| 
						 | 
					60fe48d355 | ||
| 
						 | 
					2dcd0d2b0a | ||
| 
						 | 
					8e11aa9130 | ||
| 
						 | 
					f6e223c18d | ||
| 
						 | 
					9d29b55bee | ||
| 
						 | 
					92aa8580db | ||
| 
						 | 
					538028a003 | ||
| 
						 | 
					c53575a74f | ||
| 
						 | 
					193016a46a | ||
| 
						 | 
					aaa50b4d1d | ||
| 
						 | 
					a43120320e | ||
| 
						 | 
					b8bb0c038d | ||
| 
						 | 
					dc79fc2919 | ||
| 
						 | 
					30787fef60 | ||
| 
						 | 
					445ae156ef | ||
| 
						 | 
					62a0cfb0f6 | ||
| 
						 | 
					96bc3ef99a | ||
| 
						 | 
					1d3b95d24f | ||
| 
						 | 
					56fe4b07f3 | ||
| 
						 | 
					ea60f7005b | ||
| 
						 | 
					9eb59062aa | ||
| 
						 | 
					d00927c31f | ||
| 
						 | 
					c03017208d | ||
| 
						 | 
					73f945458a | ||
| 
						 | 
					db12234611 | ||
| 
						 | 
					ed1cd4632f | ||
| 
						 | 
					17d3755152 | ||
| 
						 | 
					d7c0c2ea72 | ||
| 
						 | 
					7c823c98ae | ||
| 
						 | 
					97508a6f31 | ||
| 
						 | 
					2507a41b6e | ||
| 
						 | 
					9833accc79 | ||
| 
						 | 
					d46123771a | ||
| 
						 | 
					87fe84b1ac | ||
| 
						 | 
					21140f437e | ||
| 
						 | 
					ba9e410393 | ||
| 
						 | 
					9b628546c1 | ||
| 
						 | 
					d0837fada8 | ||
| 
						 | 
					520647d72f | ||
| 
						 | 
					51c888845c | ||
| 
						 | 
					e4606219bc | ||
| 
						 | 
					716335df2c | ||
| 
						 | 
					ad4f90c502 | ||
| 
						 | 
					a1bdfa7560 | ||
| 
						 | 
					587fb2a170 | ||
| 
						 | 
					7d801ff84c | ||
| 
						 | 
					d69accd9a5 | ||
| 
						 | 
					1127750c5e | ||
| 
						 | 
					7758bd89c1 | ||
| 
						 | 
					de7264327a | ||
| 
						 | 
					c3f0932794 | ||
| 
						 | 
					367907e037 | ||
| 
						 | 
					2d15bd651e | ||
| 
						 | 
					4b1d7863f8 | ||
| 
						 | 
					e425d768dd | ||
| 
						 | 
					34ca807044 | ||
| 
						 | 
					9075146b47 | ||
| 
						 | 
					26c4591baa | ||
| 
						 | 
					2aac8c55e7 | ||
| 
						 | 
					8af55efdb3 | ||
| 
						 | 
					9d6e07ff96 | ||
| 
						 | 
					8f58eee6af | ||
| 
						 | 
					8dd3d78f21 | ||
| 
						 | 
					48161fd02f | ||
| 
						 | 
					b61410826d | ||
| 
						 | 
					2f0188b280 | ||
| 
						 | 
					3a4fffdb0b | ||
| 
						 | 
					6393072e68 | ||
| 
						 | 
					109910d18f | ||
| 
						 | 
					8874aaabe9 | ||
| 
						 | 
					cafbea9c42 | ||
| 
						 | 
					4843ee80a7 | ||
| 
						 | 
					4511c8f30c | ||
| 
						 | 
					4cf1e52ac0 | ||
| 
						 | 
					b501b7f47c | ||
| 
						 | 
					cc275f9877 | ||
| 
						 | 
					7aae55cde7 | ||
| 
						 | 
					85eaa219c6 | ||
| 
						 | 
					7d5ecb8ba4 | ||
| 
						 | 
					1fd142d337 | ||
| 
						 | 
					d75c6aecbe | ||
| 
						 | 
					dffe0f656d | ||
| 
						 | 
					890639436b | ||
| 
						 | 
					99f66d7c5d | ||
| 
						 | 
					05faa52425 | ||
| 
						 | 
					9e1a8b646b | ||
| 
						 | 
					8f6ec03446 | ||
| 
						 | 
					c56b4fade3 | ||
| 
						 | 
					61aaaabcb5 | ||
| 
						 | 
					d57cf93580 | ||
| 
						 | 
					82ad5c103d | ||
| 
						 | 
					a0b5bc5456 | ||
| 
						 | 
					c810e541ea | ||
| 
						 | 
					05ea3b8187 | ||
| 
						 | 
					8301dffb21 | ||
| 
						 | 
					01be5243de | ||
| 
						 | 
					334196799a | ||
| 
						 | 
					c11bbcf442 | ||
| 
						 | 
					8e3a7576ea | ||
| 
						 | 
					deca6f03ba | ||
| 
						 | 
					401064d3c8 | ||
| 
						 | 
					b6f59d3c98 | ||
| 
						 | 
					1fb3663398 | ||
| 
						 | 
					5c1604e959 | ||
| 
						 | 
					17b1f3e465 | ||
| 
						 | 
					9a68bdeec1 | ||
| 
						 | 
					9b947ef734 | ||
| 
						 | 
					66432608ed | ||
| 
						 | 
					d8153ac8fc | ||
| 
						 | 
					27d9f82f7d | ||
| 
						 | 
					5b55bcd879 | ||
| 
						 | 
					5cfd28881b | ||
| 
						 | 
					bc54a42e01 | ||
| 
						 | 
					03f9964c59 | ||
| 
						 | 
					f159219d2c | ||
| 
						 | 
					e714f32737 | ||
| 
						 | 
					20858db96d | ||
| 
						 | 
					89b82bb778 | ||
| 
						 | 
					2c886d739f | ||
| 
						 | 
					1ccf4e49bc | ||
| 
						 | 
					7d63e3e088 | ||
| 
						 | 
					828523f281 | ||
| 
						 | 
					afe3831f25 | ||
| 
						 | 
					3888c56f1a | ||
| 
						 | 
					6f07966ef8 | ||
| 
						 | 
					09eafe8abd | ||
| 
						 | 
					6719a42e27 | ||
| 
						 | 
					4b98a70ee8 | ||
| 
						 | 
					db3f5447ca | ||
| 
						 | 
					fed63f645d | ||
| 
						 | 
					e7315bb570 | ||
| 
						 | 
					cd2404f26a | ||
| 
						 | 
					b866166425 | ||
| 
						 | 
					46580376dd | ||
| 
						 | 
					b8bfb44aec | ||
| 
						 | 
					a153f572d0 | ||
| 
						 | 
					66c30a59e7 | ||
| 
						 | 
					10b8efc5cb | ||
| 
						 | 
					c65d414b7b | ||
| 
						 | 
					7f7d89c745 | ||
| 
						 | 
					742028b691 | ||
| 
						 | 
					62f685bac2 | ||
| 
						 | 
					0b3333e88c | ||
| 
						 | 
					c341a99b83 | ||
| 
						 | 
					f43c420d59 | ||
| 
						 | 
					0393970a80 | ||
| 
						 | 
					1865e0661f | ||
| 
						 | 
					c07b1194b3 | ||
| 
						 | 
					bf802628b9 | ||
| 
						 | 
					36020373cd | ||
| 
						 | 
					43e73d69de | ||
| 
						 | 
					47a3f649d2 | ||
| 
						 | 
					5c63f8e52a | ||
| 
						 | 
					01c553ef13 | ||
| 
						 | 
					f229e4e12a | ||
| 
						 | 
					40cf4c8d32 | ||
| 
						 | 
					ee38c419de | ||
| 
						 | 
					10baa34c18 | ||
| 
						 | 
					343b67fa7f | ||
| 
						 | 
					6de8b4e35f | ||
| 
						 | 
					57e535c2c8 | ||
| 
						 | 
					af5b22a265 | ||
| 
						 | 
					77972c961b | ||
| 
						 | 
					a3efa5676b | ||
| 
						 | 
					014dbc2a86 | ||
| 
						 | 
					226a2941d6 | ||
| 
						 | 
					c269c8fd3f | ||
| 
						 | 
					d8fc3c1ebf | ||
| 
						 | 
					a5c6ffd1b9 | ||
| 
						 | 
					9aaaaae175 | ||
| 
						 | 
					7d39b69540 | ||
| 
						 | 
					09bad14c3d | ||
| 
						 | 
					369c9dc6e2 | ||
| 
						 | 
					9676d2cee7 | ||
| 
						 | 
					5156c67226 | ||
| 
						 | 
					4d48fc3d85 | ||
| 
						 | 
					20da329a21 | ||
| 
						 | 
					4b664cc142 | ||
| 
						 | 
					c9b620fdb2 | ||
| 
						 | 
					25c886d401 | ||
| 
						 | 
					740805356f | ||
| 
						 | 
					ce5fb57577 | ||
| 
						 | 
					3e20d2b454 | ||
| 
						 | 
					a9e8186491 | ||
| 
						 | 
					3bb909b026 | ||
| 
						 | 
					b921d91aeb | ||
| 
						 | 
					05790954c6 | ||
| 
						 | 
					13014c1351 | ||
| 
						 | 
					e34c63b830 | ||
| 
						 | 
					943100d758 | ||
| 
						 | 
					55f40d66f2 | ||
| 
						 | 
					593e5ac79c | ||
| 
						 | 
					ef31bce5ee | ||
| 
						 | 
					3c75eb96f1 | ||
| 
						 | 
					f34dfde925 | ||
| 
						 | 
					e3b72fe0aa | ||
| 
						 | 
					60de74a375 | ||
| 
						 | 
					55e58f8d35 | ||
| 
						 | 
					a465254418 | ||
| 
						 | 
					5d27a138cf | ||
| 
						 | 
					22f4b036df | ||
| 
						 | 
					03f694922d | ||
| 
						 | 
					a841e287e5 | ||
| 
						 | 
					5d2afdd825 | ||
| 
						 | 
					67240e2339 | ||
| 
						 | 
					f84a8eccfa | ||
| 
						 | 
					68a058e4f1 | ||
| 
						 | 
					d678b42ece | ||
| 
						 | 
					2cf63cda08 | ||
| 
						 | 
					7bd4eeb0df | ||
| 
						 | 
					dc3ee7c779 | ||
| 
						 | 
					e8cc97a8e5 | ||
| 
						 | 
					3b837e1d54 | ||
| 
						 | 
					bb6c2050bc | ||
| 
						 | 
					082d4f9691 | ||
| 
						 | 
					153d68a9cd | ||
| 
						 | 
					0404faa856 | ||
| 
						 | 
					afbc2d6b8f | ||
| 
						 | 
					89ecc8bd2f | ||
| 
						 | 
					7f21a2b319 | ||
| 
						 | 
					e2f07f6723 | ||
| 
						 | 
					a475e143b7 | ||
| 
						 | 
					e50fd80b2e | ||
| 
						 | 
					68ea1abc05 | ||
| 
						 | 
					2e76b306c4 | ||
| 
						 | 
					ca3cac4ed3 | ||
| 
						 | 
					41852460e1 | ||
| 
						 | 
					9ec4e083d9 | ||
| 
						 | 
					9560a1c4a7 | ||
| 
						 | 
					4f5a47ace7 | ||
| 
						 | 
					01c4d662f2 | ||
| 
						 | 
					9bdda77e89 | ||
| 
						 | 
					194024edb9 | ||
| 
						 | 
					bef0d3a6a1 | ||
| 
						 | 
					47a024b795 | ||
| 
						 | 
					39847f9c9d | ||
| 
						 | 
					fa7bd28c92 | ||
| 
						 | 
					279f78e4a8 | ||
| 
						 | 
					8ec3cbdb33 | ||
| 
						 | 
					7449f7e73f | ||
| 
						 | 
					d680fde759 | ||
| 
						 | 
					f24f21ca91 | ||
| 
						 | 
					c8ea37eec0 | ||
| 
						 | 
					b71f452795 | ||
| 
						 | 
					7ea1ece169 | ||
| 
						 | 
					aece3a37c0 | ||
| 
						 | 
					705871f8dc | ||
| 
						 | 
					4a11975349 | ||
| 
						 | 
					4d3d27f2c4 | ||
| 
						 | 
					d784a30d42 | ||
| 
						 | 
					35f776284b | ||
| 
						 | 
					f659a6fe37 | ||
| 
						 | 
					ad53c99fc4 | ||
| 
						 | 
					fa0172d00c | ||
| 
						 | 
					ba77a88714 | ||
| 
						 | 
					9b39087102 | ||
| 
						 | 
					845411b48c | ||
| 
						 | 
					d715867b09 | ||
| 
						 | 
					0ca2cdfbed | ||
| 
						 | 
					a00961b9ef | ||
| 
						 | 
					701c188bab | ||
| 
						 | 
					e81002807f | ||
| 
						 | 
					0d1c72386e | ||
| 
						 | 
					c91779dffe | ||
| 
						 | 
					3853cc9214 | ||
| 
						 | 
					a66b3f6b80 | ||
| 
						 | 
					c97ec32343 | ||
| 
						 | 
					2abba7e445 | ||
| 
						 | 
					f887c27ad1 | ||
| 
						 | 
					6ee8d74899 | ||
| 
						 | 
					f196c72563 | ||
| 
						 | 
					419e564441 | ||
| 
						 | 
					de97b54c95 | ||
| 
						 | 
					07001f7b5c | ||
| 
						 | 
					bee17fce64 | ||
| 
						 | 
					718904a853 | ||
| 
						 | 
					e14d652651 | ||
| 
						 | 
					98ae5270ef | ||
| 
						 | 
					19ccf0ab40 | ||
| 
						 | 
					72af4a69d6 | ||
| 
						 | 
					fe50f4229c | ||
| 
						 | 
					6021bec5ee | ||
| 
						 | 
					7fcadc85fa | ||
| 
						 | 
					ca4de877c1 | ||
| 
						 | 
					1dfecf9618 | ||
| 
						 | 
					0a3505ed89 | ||
| 
						 | 
					33cbf7eabe | ||
| 
						 | 
					935d97ce1a | ||
| 
						 | 
					9f73f0ca8d | ||
| 
						 | 
					5d7f971a82 | ||
| 
						 | 
					d8cdbac15e | ||
| 
						 | 
					03b8c1348c | ||
| 
						 | 
					25a0be7672 | ||
| 
						 | 
					08f1ce2d54 | ||
| 
						 | 
					bea20d0495 | ||
| 
						 | 
					c42430ccf9 | ||
| 
						 | 
					5ae10e8516 | ||
| 
						 | 
					e3f4a9ce5b | ||
| 
						 | 
					cf1fb606fb | ||
| 
						 | 
					54ec81b67d | ||
| 
						 | 
					f2a9725572 | ||
| 
						 | 
					4765114e80 | ||
| 
						 | 
					5ff757ad65 | ||
| 
						 | 
					1642c68493 | ||
| 
						 | 
					f31f10cea9 | ||
| 
						 | 
					76e0bbb55d | ||
| 
						 | 
					f43af9c0a5 | ||
| 
						 | 
					f7a3d2705c | ||
| 
						 | 
					22c8af0cc5 | ||
| 
						 | 
					f263a5221d | ||
| 
						 | 
					3834ab8ede | ||
| 
						 | 
					e2e167630d | ||
| 
						 | 
					01dd44300b | ||
| 
						 | 
					b30160d671 | ||
| 
						 | 
					f44d505b41 | ||
| 
						 | 
					16fa6904d9 | ||
| 
						 | 
					b58c17e75e | ||
| 
						 | 
					ae590d42dc | ||
| 
						 | 
					d7917160c0 | ||
| 
						 | 
					01e4414d17 | ||
| 
						 | 
					0bc2eb530d | ||
| 
						 | 
					12b124e5a3 | ||
| 
						 | 
					478a4b2593 | ||
| 
						 | 
					9752e30eb4 | ||
| 
						 | 
					af6e87ba31 | ||
| 
						 | 
					64d390ad0f | ||
| 
						 | 
					c94bcb6896 | ||
| 
						 | 
					97f9df2f2d | ||
| 
						 | 
					4e7f68a86c | ||
| 
						 | 
					2f7f677549 | ||
| 
						 | 
					12a8a1531d | ||
| 
						 | 
					f44d867d3a | ||
| 
						 | 
					6f636187f7 | ||
| 
						 | 
					9414f89e50 | ||
| 
						 | 
					60bf1a5451 | ||
| 
						 | 
					32ba8f4731 | ||
| 
						 | 
					81f96de2bd | ||
| 
						 | 
					0c417755ed | ||
| 
						 | 
					93e5bde797 | ||
| 
						 | 
					c85f69c9ee | ||
| 
						 | 
					b6eaf0a7c5 | ||
| 
						 | 
					5f1851bade | ||
| 
						 | 
					5c66a02711 | ||
| 
						 | 
					bde925a0e3 | ||
| 
						 | 
					0f574a765b | ||
| 
						 | 
					782b941531 | ||
| 
						 | 
					f42c0a0717 | ||
| 
						 | 
					13ac14d449 | ||
| 
						 | 
					db9cea81db | ||
| 
						 | 
					7c1fd542da | ||
| 
						 | 
					54a2b2534a | ||
| 
						 | 
					f5fb6c1e03 | ||
| 
						 | 
					781c0701fc | ||
| 
						 | 
					742f1f85dc | ||
| 
						 | 
					a648e9be49 | ||
| 
						 | 
					fd9441dde2 | ||
| 
						 | 
					b5ec59c396 | ||
| 
						 | 
					60e4594abd | ||
| 
						 | 
					79692ef58a | ||
| 
						 | 
					ace7ee5622 | ||
| 
						 | 
					741ac679a0 | ||
| 
						 | 
					216526e391 | ||
| 
						 | 
					d76af2cb61 | ||
| 
						 | 
					b7d4c40736 | ||
| 
						 | 
					6092af8de6 | ||
| 
						 | 
					627424b8b9 | ||
| 
						 | 
					e33aff7cf3 | ||
| 
						 | 
					ef0bfb237a | ||
| 
						 | 
					c042c5568b | ||
| 
						 | 
					d84a7ee358 | ||
| 
						 | 
					8bfc8ece9d | ||
| 
						 | 
					2d3cf7d84d | ||
| 
						 | 
					520ef8f1df | ||
| 
						 | 
					f251d4267f | ||
| 
						 | 
					2052a5351c | ||
| 
						 | 
					9807d0aede | ||
| 
						 | 
					a41afcd714 | ||
| 
						 | 
					d93d2b5945 | ||
| 
						 | 
					d54a129605 | ||
| 
						 | 
					77911980cb | ||
| 
						 | 
					d51fd1e2f9 | ||
| 
						 | 
					fe54f8eb16 | ||
| 
						 | 
					fc7c4af27a | ||
| 
						 | 
					09e7600d86 | ||
| 
						 | 
					17410874e3 | ||
| 
						 | 
					03d4174163 | ||
| 
						 | 
					99eff73b0d | ||
| 
						 | 
					acefa39796 | ||
| 
						 | 
					c01c0528a6 | ||
| 
						 | 
					0ec58007c9 | ||
| 
						 | 
					e8daf88729 | ||
| 
						 | 
					ab74c7f7eb | ||
| 
						 | 
					6b673c7f44 | ||
| 
						 | 
					53510a3cb9 | ||
| 
						 | 
					d4d38a880d | ||
| 
						 | 
					18783d5e3b | ||
| 
						 | 
					eb235cb552 | ||
| 
						 | 
					435a6b6d53 | ||
| 
						 | 
					8d13745c6b | ||
| 
						 | 
					14c7cfc64c | ||
| 
						 | 
					c7821b9cee | ||
| 
						 | 
					a1d66aef0c | ||
| 
						 | 
					e275f1f4b9 | ||
| 
						 | 
					48de8b0739 | ||
| 
						 | 
					b75dc0efe0 | ||
| 
						 | 
					1d498349c5 | ||
| 
						 | 
					311e1cfb00 | ||
| 
						 | 
					5cdcec699b | ||
| 
						 | 
					cd72287d99 | ||
| 
						 | 
					c8717bfa32 | ||
| 
						 | 
					83de75b689 | ||
| 
						 | 
					e5ea762cbc | ||
| 
						 | 
					01df01cd66 | ||
| 
						 | 
					2c07a2c825 | ||
| 
						 | 
					c3f50ba0fb | ||
| 
						 | 
					c04419fd09 | ||
| 
						 | 
					9c7af0dfce | ||
| 
						 | 
					b66d14e980 | ||
| 
						 | 
					6a553e9554 | ||
| 
						 | 
					4273b72d71 | ||
| 
						 | 
					9ccfa79199 | ||
| 
						 | 
					fe3d22d4f8 | ||
| 
						 | 
					e06642e892 | ||
| 
						 | 
					5199e946a1 | ||
| 
						 | 
					17aff2f9b8 | ||
| 
						 | 
					f7c7ac44f7 | ||
| 
						 | 
					62dd0a561e | ||
| 
						 | 
					858eacddea | ||
| 
						 | 
					471bb5169c | ||
| 
						 | 
					9d89aa329c | ||
| 
						 | 
					4e4d8bdc5e | ||
| 
						 | 
					a30ec32ac1 | ||
| 
						 | 
					a9192ae2e1 | ||
| 
						 | 
					3d4a0b02e5 | ||
| 
						 | 
					3659cf7c87 | ||
| 
						 | 
					fbcf35414c | ||
| 
						 | 
					d79e5dd8fb | ||
| 
						 | 
					92b116c0da | ||
| 
						 | 
					da3f911deb | ||
| 
						 | 
					9d82ce8ab4 | ||
| 
						 | 
					db9597d2e7 | ||
| 
						 | 
					1523558f4c | ||
| 
						 | 
					bd9f4fe41c | ||
| 
						 | 
					6b938b2597 | ||
| 
						 | 
					5a8009e46e | ||
| 
						 | 
					c92eeb6bb7 | ||
| 
						 | 
					8ea6baaf5d | ||
| 
						 | 
					1ed03842c0 | ||
| 
						 | 
					0e9984413c | ||
| 
						 | 
					362b419814 | ||
| 
						 | 
					bffcccc1fe | ||
| 
						 | 
					b8e9a4ce9f | ||
| 
						 | 
					bdff3fd452 | ||
| 
						 | 
					1fc51f0087 | ||
| 
						 | 
					226013d999 | ||
| 
						 | 
					86847263b8 | ||
| 
						 | 
					b798523d53 | ||
| 
						 | 
					bd59c4fccf | ||
| 
						 | 
					566ffe24a4 | ||
| 
						 | 
					0e5c1b2041 | ||
| 
						 | 
					9a088a21da | ||
| 
						 | 
					1160d27004 | ||
| 
						 | 
					b4e5740050 | ||
| 
						 | 
					12bb3f5796 | ||
| 
						 | 
					ff62fdb69d | ||
| 
						 | 
					4ebf32cb1f | ||
| 
						 | 
					5afb8a77a9 | ||
| 
						 | 
					48ed33af95 | ||
| 
						 | 
					4a64cd4464 | ||
| 
						 | 
					8ae1a1b558 | ||
| 
						 | 
					ef1dd8b761 | ||
| 
						 | 
					3766f44787 | ||
| 
						 | 
					101067d018 | ||
| 
						 | 
					448d19bfbb | ||
| 
						 | 
					c1caad6d43 | ||
| 
						 | 
					a653bf5b0d | ||
| 
						 | 
					afdb369e04 | ||
| 
						 | 
					f02841409c | ||
| 
						 | 
					d4000cf662 | ||
| 
						 | 
					e1cf5919fa | ||
| 
						 | 
					d33e8d77c2 | ||
| 
						 | 
					3a522215a9 | ||
| 
						 | 
					7de6ea0879 | ||
| 
						 | 
					5ee0250ba5 | ||
| 
						 | 
					69d0a22091 | ||
| 
						 | 
					4d6c11ce31 | ||
| 
						 | 
					178605664e | ||
| 
						 | 
					12acb5473b | ||
| 
						 | 
					0cf8004b8d | ||
| 
						 | 
					6b0a8eae74 | ||
| 
						 | 
					00412c7216 | ||
| 
						 | 
					6483f23558 | ||
| 
						 | 
					4234d51a29 | ||
| 
						 | 
					9243d300cc | ||
| 
						 | 
					2ce70206c6 | ||
| 
						 | 
					b22455d2a5 | ||
| 
						 | 
					4f0bb9f6c3 | ||
| 
						 | 
					7dfa1b0942 | ||
| 
						 | 
					093e23c006 | ||
| 
						 | 
					5f003ccbe2 | ||
| 
						 | 
					8d0608610f | ||
| 
						 | 
					8bfe583a20 | ||
| 
						 | 
					0a3172dfdb | ||
| 
						 | 
					5c0e151bc2 | ||
| 
						 | 
					e69f36047d | ||
| 
						 | 
					ed368ddd9d | ||
| 
						 | 
					8400e90e34 | ||
| 
						 | 
					4cd95b724b | ||
| 
						 | 
					46b3836fbd | ||
| 
						 | 
					9988227d93 | ||
| 
						 | 
					9d289bfa34 | ||
| 
						 | 
					048de6b388 | ||
| 
						 | 
					9b7d8934da | ||
| 
						 | 
					0dd9b21c2d | ||
| 
						 | 
					417184525f | ||
| 
						 | 
					0a09ec706f | ||
| 
						 | 
					cf43b26e14 | ||
| 
						 | 
					a8b27e224f | ||
| 
						 | 
					dc5d14834d | ||
| 
						 | 
					12c935f647 | ||
| 
						 | 
					7710cb245c | ||
| 
						 | 
					a6b77c0457 | ||
| 
						 | 
					748a05f355 | ||
| 
						 | 
					831b9da0cf | ||
| 
						 | 
					a3339c9d5f | ||
| 
						 | 
					d228f38471 | ||
| 
						 | 
					fe13853b8b | ||
| 
						 | 
					6f4dbdc959 | ||
| 
						 | 
					870f0bcbb1 | ||
| 
						 | 
					599dd81e3c | ||
| 
						 | 
					9e99d158fd | ||
| 
						 | 
					136ebb5a07 | ||
| 
						 | 
					707338b1aa | ||
| 
						 | 
					7e06bd53b6 | ||
| 
						 | 
					08c1b864fc | ||
| 
						 | 
					16e7a16d12 | ||
| 
						 | 
					45200da32f | ||
| 
						 | 
					84a9ca59ef | ||
| 
						 | 
					40f4c35b42 | ||
| 
						 | 
					bb77d34017 | ||
| 
						 | 
					9659ebe59b | ||
| 
						 | 
					16c914b139 | ||
| 
						 | 
					142f26add1 | ||
| 
						 | 
					ef7d2aea8d | ||
| 
						 | 
					1aa40cb6df | ||
| 
						 | 
					cd06b931a9 | ||
| 
						 | 
					fd2df92000 | ||
| 
						 | 
					b565a8b8f7 | ||
| 
						 | 
					1aab656705 | ||
| 
						 | 
					6919d0cde6 | ||
| 
						 | 
					95fd8de1bc | ||
| 
						 | 
					28ca1a5193 | ||
| 
						 | 
					a1d07e5a00 | ||
| 
						 | 
					131a7f3782 | ||
| 
						 | 
					c28a7d6c10 | ||
| 
						 | 
					6b20bb967b | ||
| 
						 | 
					75f228418d | ||
| 
						 | 
					41f8b0d19b | ||
| 
						 | 
					cea3b8b010 | ||
| 
						 | 
					aba0e1f026 | ||
| 
						 | 
					f42587af22 | ||
| 
						 | 
					edcb7e87bb | ||
| 
						 | 
					96f0ceeb8c | ||
| 
						 | 
					a9baa7f1c1 | ||
| 
						 | 
					b1483287dc | ||
| 
						 | 
					a4657541fc | ||
| 
						 | 
					56d88b4c56 | ||
| 
						 | 
					02313e4be8 | ||
| 
						 | 
					2a14a3a4dc | ||
| 
						 | 
					d02a2e8c2e | ||
| 
						 | 
					0d281f8437 | ||
| 
						 | 
					2b0f43f334 | ||
| 
						 | 
					f9d28fc124 | ||
| 
						 | 
					da07173471 | ||
| 
						 | 
					4deeff7029 | ||
| 
						 | 
					1538fbb102 | ||
| 
						 | 
					b9259b87eb | ||
| 
						 | 
					30997dbc88 | ||
| 
						 | 
					a9d926e80a | ||
| 
						 | 
					c41369c89c | ||
| 
						 | 
					656bef3da9 | ||
| 
						 | 
					607eb6d130 | ||
| 
						 | 
					f2e9b3577d | ||
| 
						 | 
					cb2c6d8560 | ||
| 
						 | 
					bfe8346ced | ||
| 
						 | 
					88da9bb91b | ||
| 
						 | 
					5e2ee1a16c | ||
| 
						 | 
					2fdc746392 | ||
| 
						 | 
					1667973a66 | ||
| 
						 | 
					42f0101440 | ||
| 
						 | 
					13b69bff1b | ||
| 
						 | 
					2c2226dfd6 | ||
| 
						 | 
					a3fdfe0e15 | ||
| 
						 | 
					a0de209a55 | ||
| 
						 | 
					0208b50ac7 | ||
| 
						 | 
					4a61779aba | ||
| 
						 | 
					05057ade05 | ||
| 
						 | 
					6fb206853c | ||
| 
						 | 
					fab68055bf | ||
| 
						 | 
					d5a77ef3cd | ||
| 
						 | 
					dcb2605de4 | ||
| 
						 | 
					e6d38f4539 | ||
| 
						 | 
					293b56cfa6 | ||
| 
						 | 
					775e93d54b | ||
| 
						 | 
					7f840e75df | ||
| 
						 | 
					2113ea675e | ||
| 
						 | 
					916a5c1a6b | ||
| 
						 | 
					f684531315 | ||
| 
						 | 
					fe8dda8996 | ||
| 
						 | 
					4cd4b328c8 | ||
| 
						 | 
					3d7ee6a4df | ||
| 
						 | 
					d844c89b94 | ||
| 
						 | 
					177ea2b85a | ||
| 
						 | 
					50c5c15f49 | ||
| 
						 | 
					1810760dc7 | ||
| 
						 | 
					4635b92e3f | ||
| 
						 | 
					1c652626eb | ||
| 
						 | 
					2000cfb1db | ||
| 
						 | 
					f4d07828e7 | ||
| 
						 | 
					95b552671c | ||
| 
						 | 
					ef3bc3efe1 | ||
| 
						 | 
					371ad899f5 | ||
| 
						 | 
					2c54158d84 | ||
| 
						 | 
					5d9e30bbdc | ||
| 
						 | 
					e477fd567d | ||
| 
						 | 
					6a6c2937fe | ||
| 
						 | 
					09e17c4da8 | ||
| 
						 | 
					fd00469d11 | ||
| 
						 | 
					cbbeb795f3 | ||
| 
						 | 
					bba40e0da8 | ||
| 
						 | 
					d23165d06a | ||
| 
						 | 
					405fef6f03 | ||
| 
						 | 
					588f217826 | ||
| 
						 | 
					3d8b7cf80e | ||
| 
						 | 
					c0ef923ad3 | ||
| 
						 | 
					3df44fc71e | ||
| 
						 | 
					c1965492d9 | ||
| 
						 | 
					1f56ffde80 | ||
| 
						 | 
					f335fdc002 | ||
| 
						 | 
					0c914b5ec8 | ||
| 
						 | 
					d767b06858 | ||
| 
						 | 
					d4e49f3944 | ||
| 
						 | 
					7dfc5b3faf | ||
| 
						 | 
					8a88033ab9 | ||
| 
						 | 
					7b06b38c94 | ||
| 
						 | 
					5409752817 | ||
| 
						 | 
					909f3a3005 | ||
| 
						 | 
					4930532c7b | ||
| 
						 | 
					8a42e65c6a | ||
| 
						 | 
					5d4121a9b4 | ||
| 
						 | 
					a70e6c49a1 | ||
| 
						 | 
					3d83d5f4b5 | ||
| 
						 | 
					f9dece0743 | ||
| 
						 | 
					ac0871d0e8 | ||
| 
						 | 
					ffc19e591d | ||
| 
						 | 
					c53380ca3d | ||
| 
						 | 
					7c74a2026a | ||
| 
						 | 
					adaed438d9 | ||
| 
						 | 
					baf38305cb | ||
| 
						 | 
					8254712521 | ||
| 
						 | 
					53214781e3 | ||
| 
						 | 
					88cbbbdf65 | ||
| 
						 | 
					c10dca9c7b | ||
| 
						 | 
					7f2ebb4bde | ||
| 
						 | 
					f1abb60e4a | ||
| 
						 | 
					e014c7aff6 | ||
| 
						 | 
					b79c03433e | ||
| 
						 | 
					34eb4d974d | ||
| 
						 | 
					3264be3c5e | ||
| 
						 | 
					655f4f75fb | ||
| 
						 | 
					4383f31696 | ||
| 
						 | 
					99eb15d15e | ||
| 
						 | 
					2682c6e150 | ||
| 
						 | 
					3a5d854e6d | ||
| 
						 | 
					1e90c6387c | ||
| 
						 | 
					2cca25f4d0 | ||
| 
						 | 
					565724d201 | ||
| 
						 | 
					3e4955becd | ||
| 
						 | 
					7b560c727f | ||
| 
						 | 
					35abd9dfdb | ||
| 
						 | 
					b7ccf3e0e5 | ||
| 
						 | 
					0d9ab8fdd0 | ||
| 
						 | 
					303f9290a8 | ||
| 
						 | 
					e0c4dc08a1 | ||
| 
						 | 
					8c655883fe | ||
| 
						 | 
					ba90785115 | ||
| 
						 | 
					7b392b626b | ||
| 
						 | 
					8e4ceb7d48 | ||
| 
						 | 
					2ab1c6e9a9 | ||
| 
						 | 
					dbdced0971 | ||
| 
						 | 
					5e481880bd | ||
| 
						 | 
					faec063f34 | ||
| 
						 | 
					bbea38d227 | ||
| 
						 | 
					a0ef60de49 | ||
| 
						 | 
					3313572606 | ||
| 
						 | 
					c4f850cb14 | ||
| 
						 | 
					3bdab738c6 | ||
| 
						 | 
					faaef31b9f | ||
| 
						 | 
					ca7b8b8b4c | ||
| 
						 | 
					9ca84e0694 | ||
| 
						 | 
					daaf2b1796 | ||
| 
						 | 
					25f7cbea5a | ||
| 
						 | 
					c485ea9d7b | ||
| 
						 | 
					295390c8e9 | ||
| 
						 | 
					3ebf816ce2 | ||
| 
						 | 
					0e362b851b | ||
| 
						 | 
					8d7ba19a08 | ||
| 
						 | 
					08f4aa9d10 | ||
| 
						 | 
					98175d5c72 | ||
| 
						 | 
					7d4cad90bc | ||
| 
						 | 
					335354d962 | ||
| 
						 | 
					fe31d15d27 | ||
| 
						 | 
					7ceb6eb50d | ||
| 
						 | 
					4c4db46aa8 | ||
| 
						 | 
					b5724ed343 | ||
| 
						 | 
					cae94175fe | ||
| 
						 | 
					0494a9d410 | ||
| 
						 | 
					c261b5c1ce | ||
| 
						 | 
					c89e17ac00 | ||
| 
						 | 
					50e7410002 | ||
| 
						 | 
					c5b0ebf76d | ||
| 
						 | 
					1d08978d6c | ||
| 
						 | 
					fc78b6c933 | ||
| 
						 | 
					480a5718fc | ||
| 
						 | 
					f093bd115c | ||
| 
						 | 
					8a86beff14 | ||
| 
						 | 
					6020890384 | ||
| 
						 | 
					124aa947e2 | ||
| 
						 | 
					e1add14453 | ||
| 
						 | 
					e3293837a8 | ||
| 
						 | 
					5cb2743780 | ||
| 
						 | 
					6f0c79ec25 | ||
| 
						 | 
					7de7d1d926 | ||
| 
						 | 
					89175f8e85 | ||
| 
						 | 
					fc48c59eb0 | ||
| 
						 | 
					51332bc7e7 | ||
| 
						 | 
					7403405d12 | ||
| 
						 | 
					1d13947e71 | ||
| 
						 | 
					f6cb1ffe20 | ||
| 
						 | 
					6d92b5651a | ||
| 
						 | 
					3ea5bb2a6c | ||
| 
						 | 
					1d367eca69 | ||
| 
						 | 
					d4bf3a2ec3 | ||
| 
						 | 
					0ef8881660 | ||
| 
						 | 
					d7d1121f7d | ||
| 
						 | 
					7f089c309f | ||
| 
						 | 
					4dcc0bb66c | ||
| 
						 | 
					0049be7feb | ||
| 
						 | 
					39ff641be9 | ||
| 
						 | 
					e2fed24995 | ||
| 
						 | 
					c0aa353f83 | ||
| 
						 | 
					d8521be63d | ||
| 
						 | 
					6d4569c89d | ||
| 
						 | 
					cd07553b59 | ||
| 
						 | 
					641bfcc9f7 | ||
| 
						 | 
					6c01371958 | ||
| 
						 | 
					7b00260b1a | ||
| 
						 | 
					875142373e | ||
| 
						 | 
					ba505b15ef | ||
| 
						 | 
					17d227b142 | ||
| 
						 | 
					e7e192ffe3 | ||
| 
						 | 
					c53ec6e12d | ||
| 
						 | 
					aad6492a6a | ||
| 
						 | 
					fd5b125c2d | ||
| 
						 | 
					5acee76c70 | ||
| 
						 | 
					10916fa82a | ||
| 
						 | 
					f69951a523 | ||
| 
						 | 
					38ba85e89d | ||
| 
						 | 
					97023921b8 | ||
| 
						 | 
					f835810f0a | ||
| 
						 | 
					46f5589530 | ||
| 
						 | 
					ff9840c8ef | ||
| 
						 | 
					0c197558a1 | ||
| 
						 | 
					c409ba149d | ||
| 
						 | 
					0b896ddfb1 | ||
| 
						 | 
					45721eb4fe | ||
| 
						 | 
					1289bd03b2 | ||
| 
						 | 
					c1ba8ba6b8 | ||
| 
						 | 
					4973d8f629 | ||
| 
						 | 
					3aff4c96c4 | ||
| 
						 | 
					4005bc8985 | ||
| 
						 | 
					62e9792c39 | ||
| 
						 | 
					33183cc595 | ||
| 
						 | 
					394d552856 | ||
| 
						 | 
					aa4f0929e0 | ||
| 
						 | 
					f99b9215e3 | ||
| 
						 | 
					c51d621fee | ||
| 
						 | 
					7499892bc2 | ||
| 
						 | 
					3fb35871c7 | ||
| 
						 | 
					d6d20cd704 | ||
| 
						 | 
					9cc6a6b885 | ||
| 
						 | 
					ee0be7b6d0 | ||
| 
						 | 
					7e6153ba7d | 
@@ -4,7 +4,7 @@
 | 
			
		||||
    "dockerfile": "Dockerfile",
 | 
			
		||||
    "context": ".."
 | 
			
		||||
  },
 | 
			
		||||
  "appPort": 8123,
 | 
			
		||||
  "appPort": "8124:8123",
 | 
			
		||||
  "context": "..",
 | 
			
		||||
  "postCreateCommand": "script/bootstrap",
 | 
			
		||||
  "extensions": [
 | 
			
		||||
@@ -26,6 +26,9 @@
 | 
			
		||||
    "[typescript]": {
 | 
			
		||||
      "editor.defaultFormatter": "esbenp.prettier-vscode"
 | 
			
		||||
    },
 | 
			
		||||
    "[javascript]": {
 | 
			
		||||
      "editor.defaultFormatter": "esbenp.prettier-vscode"
 | 
			
		||||
    },
 | 
			
		||||
    "files.trimTrailingWhitespace": true
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -4,8 +4,7 @@
 | 
			
		||||
    "plugin:@typescript-eslint/recommended",
 | 
			
		||||
    "plugin:wc/recommended",
 | 
			
		||||
    "plugin:lit/recommended",
 | 
			
		||||
    "prettier",
 | 
			
		||||
    "prettier/@typescript-eslint"
 | 
			
		||||
    "prettier"
 | 
			
		||||
  ],
 | 
			
		||||
  "parser": "@typescript-eslint/parser",
 | 
			
		||||
  "parserOptions": {
 | 
			
		||||
@@ -29,9 +28,7 @@
 | 
			
		||||
    "__BUILD__": false,
 | 
			
		||||
    "__VERSION__": false,
 | 
			
		||||
    "__STATIC_PATH__": false,
 | 
			
		||||
    "Polymer": true,
 | 
			
		||||
    "webkitSpeechRecognition": false,
 | 
			
		||||
    "ResizeObserver": false
 | 
			
		||||
    "Polymer": true
 | 
			
		||||
  },
 | 
			
		||||
  "env": {
 | 
			
		||||
    "browser": true,
 | 
			
		||||
@@ -84,8 +81,29 @@
 | 
			
		||||
    "@typescript-eslint/no-unused-vars": 0,
 | 
			
		||||
    "@typescript-eslint/explicit-function-return-type": 0,
 | 
			
		||||
    "@typescript-eslint/explicit-module-boundary-types": 0,
 | 
			
		||||
    "@typescript-eslint/no-shadow": ["error"]
 | 
			
		||||
    "@typescript-eslint/no-shadow": ["error"],
 | 
			
		||||
    "@typescript-eslint/naming-convention": [
 | 
			
		||||
      0,
 | 
			
		||||
      {
 | 
			
		||||
        "selector": "default",
 | 
			
		||||
        "format": ["camelCase", "snake_case"],
 | 
			
		||||
        "leadingUnderscore": "allow",
 | 
			
		||||
        "trailingUnderscore": "allow"
 | 
			
		||||
      },
 | 
			
		||||
      {
 | 
			
		||||
        "selector": ["variable"],
 | 
			
		||||
        "format": ["camelCase", "snake_case", "UPPER_CASE"],
 | 
			
		||||
        "leadingUnderscore": "allow",
 | 
			
		||||
        "trailingUnderscore": "allow"
 | 
			
		||||
      },
 | 
			
		||||
      {
 | 
			
		||||
        "selector": "typeLike",
 | 
			
		||||
        "format": ["PascalCase"]
 | 
			
		||||
      }
 | 
			
		||||
    ],
 | 
			
		||||
    "lit/attribute-value-entities": 0
 | 
			
		||||
  },
 | 
			
		||||
  "plugins": ["disable", "import", "lit", "prettier", "@typescript-eslint"],
 | 
			
		||||
  "processor": "disable/disable"
 | 
			
		||||
  "processor": "disable/disable",
 | 
			
		||||
  "ignorePatterns": ["src/resources/lit-virtualizer/*"]
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -74,12 +74,12 @@ DO NOT DELETE ANY TEXT from this template! Otherwise, your issue may be closed w
 | 
			
		||||
 | 
			
		||||
```
 | 
			
		||||
 | 
			
		||||
## Problem-relevant configuration
 | 
			
		||||
## Problem-relevant frontend configuration
 | 
			
		||||
 | 
			
		||||
<!--
 | 
			
		||||
  An example configuration that caused the problem for you. Fill this out even
 | 
			
		||||
  if it seems unimportant to you. Please be sure to remove personal information
 | 
			
		||||
  like passwords, private URLs and other credentials.
 | 
			
		||||
  An example configuration that caused the problem for you, e.g. the YAML configuration
 | 
			
		||||
  of the used cards. Fill this out even if it seems unimportant to you. Please be sure
 | 
			
		||||
  to remove personal information like passwords, private URLs and other credentials.
 | 
			
		||||
-->
 | 
			
		||||
 | 
			
		||||
```yaml
 | 
			
		||||
@@ -89,7 +89,7 @@ DO NOT DELETE ANY TEXT from this template! Otherwise, your issue may be closed w
 | 
			
		||||
## Javascript errors shown in your browser console/inspector
 | 
			
		||||
 | 
			
		||||
<!--
 | 
			
		||||
  If you come across any javascript or other error logs, e.g., in your browser
 | 
			
		||||
  If you come across any Javascript or other error logs, e.g. in your browser
 | 
			
		||||
  console/inspector please provide them.
 | 
			
		||||
-->
 | 
			
		||||
 | 
			
		||||
							
								
								
									
										121
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										121
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,121 @@
 | 
			
		||||
name: Report a bug with the UI, Frontend or Lovelace
 | 
			
		||||
description: Report an issue related to the Home Assistant frontend.
 | 
			
		||||
labels: bug
 | 
			
		||||
body:
 | 
			
		||||
  - type: markdown
 | 
			
		||||
    attributes:
 | 
			
		||||
      value: |
 | 
			
		||||
        Make sure you are running the [latest version of Home Assistant][releases] before reporting an issue.
 | 
			
		||||
 | 
			
		||||
        If you have a feature or enhancement request for the frontend, please [start an discussion][fr] instead of creating an issue.
 | 
			
		||||
 | 
			
		||||
        **Please not not report issues for custom Lovelace cards.**
 | 
			
		||||
 | 
			
		||||
        [fr]: https://github.com/home-assistant/frontend/discussions
 | 
			
		||||
        [releases]: https://github.com/home-assistant/home-assistant/releases
 | 
			
		||||
  - type: checkboxes
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: Checklist
 | 
			
		||||
      description: Please verify that you've followed these steps
 | 
			
		||||
      options:
 | 
			
		||||
        - label: I have updated to the latest available Home Assistant version.
 | 
			
		||||
          required: true
 | 
			
		||||
        - label: I have cleared the cache of my browser.
 | 
			
		||||
          required: true
 | 
			
		||||
        - label: I have tried a different browser to see if it is related to my browser.
 | 
			
		||||
          required: true
 | 
			
		||||
  - type: markdown
 | 
			
		||||
    attributes:
 | 
			
		||||
      value: |
 | 
			
		||||
        ## The problem
 | 
			
		||||
  - type: textarea
 | 
			
		||||
    validations:
 | 
			
		||||
      required: true
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: Describe the issue you are experiencing
 | 
			
		||||
      description: Provide a clear and concise description of what the bug is.
 | 
			
		||||
  - type: textarea
 | 
			
		||||
    validations:
 | 
			
		||||
      required: true
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: Describe the behavior you expected
 | 
			
		||||
      description: Describe what you expected to happen or it should look/behave.
 | 
			
		||||
  - type: textarea
 | 
			
		||||
    validations:
 | 
			
		||||
      required: true
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: Steps to reproduce the issue
 | 
			
		||||
      description: |
 | 
			
		||||
        Please tell us exactly how to reproduce your issue.
 | 
			
		||||
        Provide clear and concise step by step instructions and add code snippets if needed.
 | 
			
		||||
      value: |
 | 
			
		||||
        1.
 | 
			
		||||
        2.
 | 
			
		||||
        3.
 | 
			
		||||
        ...
 | 
			
		||||
  - type: markdown
 | 
			
		||||
    attributes:
 | 
			
		||||
      value: |
 | 
			
		||||
        ## Environment
 | 
			
		||||
  - type: input
 | 
			
		||||
    validations:
 | 
			
		||||
      required: true
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: What version of Home Assistant Core has the issue?
 | 
			
		||||
      placeholder: core-
 | 
			
		||||
      description: >
 | 
			
		||||
        Can be found in the Configuration panel -> Info.
 | 
			
		||||
  - type: input
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: What was the last working version of Home Assistant Core?
 | 
			
		||||
      placeholder: core-
 | 
			
		||||
      description: >
 | 
			
		||||
        If known, otherwise leave blank.
 | 
			
		||||
  - type: input
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: In which browser are you experiencing the issue with?
 | 
			
		||||
      placeholder: Google Chrome 88.0.4324.150
 | 
			
		||||
      description: >
 | 
			
		||||
        Provide the full name and don't forget to add the version!
 | 
			
		||||
  - type: input
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: Which operating system are you using to run this browser?
 | 
			
		||||
      placeholder: macOS Big Sur (1.11)
 | 
			
		||||
      description: >
 | 
			
		||||
        Don't forget to add the version!
 | 
			
		||||
  - type: markdown
 | 
			
		||||
    attributes:
 | 
			
		||||
      value: |
 | 
			
		||||
        # Details
 | 
			
		||||
 | 
			
		||||
  - type: textarea
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: State of relevant entities
 | 
			
		||||
      description: >
 | 
			
		||||
        If your issue is about how an entity is shown in the UI, please add the
 | 
			
		||||
        state and attributes for all situations. You can find this information
 | 
			
		||||
        at Developer Tools -> States.
 | 
			
		||||
      render: txt
 | 
			
		||||
  - type: textarea
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: Problem-relevant frontend configuration
 | 
			
		||||
      description: >
 | 
			
		||||
        An example configuration that caused the problem for you, e.g., the YAML
 | 
			
		||||
        configuration of the used cards. Fill this out even if it seems
 | 
			
		||||
        unimportant to you. Please be sure to remove personal information like
 | 
			
		||||
        passwords, private URLs and other credentials.
 | 
			
		||||
      render: yaml
 | 
			
		||||
  - type: textarea
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: Javascript errors shown in your browser console/inspector
 | 
			
		||||
      description: >
 | 
			
		||||
        If you come across any Javascript or other error logs, e.g., in your
 | 
			
		||||
        browser console/inspector please provide them.
 | 
			
		||||
      render: txt
 | 
			
		||||
  - type: textarea
 | 
			
		||||
    attributes:
 | 
			
		||||
      label: Additional information
 | 
			
		||||
      description: >
 | 
			
		||||
        If you have any additional information for us, use the field below.
 | 
			
		||||
        Please note, you can attach screenshots or screen recordings here, by
 | 
			
		||||
        dragging and dropping files in the field below.
 | 
			
		||||
							
								
								
									
										6
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							@@ -37,9 +37,11 @@ jobs:
 | 
			
		||||
      - name: Build resources
 | 
			
		||||
        run: ./node_modules/.bin/gulp gen-icons-json build-translations gather-gallery-demos
 | 
			
		||||
      - name: Run eslint
 | 
			
		||||
        run: ./node_modules/.bin/eslint '{**/src,src}/**/*.{js,ts,html}' --ignore-path .gitignore
 | 
			
		||||
        run: yarn run lint:eslint
 | 
			
		||||
      - name: Run tsc
 | 
			
		||||
        run: ./node_modules/.bin/tsc
 | 
			
		||||
        run: yarn run lint:types
 | 
			
		||||
      - name: Run prettier
 | 
			
		||||
        run: yarn run lint:prettier
 | 
			
		||||
  test:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										19
									
								
								.github/workflows/netflify.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								.github/workflows/netflify.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
name: Netlify
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  schedule:
 | 
			
		||||
    - cron: "0 0 * * *"
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  trigger_builds:
 | 
			
		||||
    name: Trigger netlify build preview
 | 
			
		||||
    runs-on: "ubuntu-latest"
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Trigger Cast build
 | 
			
		||||
        run: curl -X POST -d {} https://api.netlify.com/build_hooks/${{ secrets.NETLIFY_CAST_DEV_BUILD_HOOK }}
 | 
			
		||||
 | 
			
		||||
      - 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 }}
 | 
			
		||||
							
								
								
									
										83
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,83 @@
 | 
			
		||||
name: Release
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  release:
 | 
			
		||||
    types:
 | 
			
		||||
      - published
 | 
			
		||||
 | 
			
		||||
env:
 | 
			
		||||
  PYTHON_VERSION: 3.8
 | 
			
		||||
  NODE_VERSION: 12.1
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  release:
 | 
			
		||||
    name: Release
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Checkout the repository
 | 
			
		||||
        uses: actions/checkout@v2
 | 
			
		||||
 | 
			
		||||
      - name: Verify version
 | 
			
		||||
        uses: home-assistant/actions/helpers/verify-version@master
 | 
			
		||||
 | 
			
		||||
      - name: Set up Python ${{ env.PYTHON_VERSION }}
 | 
			
		||||
        uses: actions/setup-python@v2
 | 
			
		||||
        with:
 | 
			
		||||
          python-version: ${{ env.PYTHON_VERSION }}
 | 
			
		||||
 | 
			
		||||
      - name: Set up Node ${{ env.NODE_VERSION }}
 | 
			
		||||
        uses: actions/setup-node@v2
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: ${{ env.NODE_VERSION }}
 | 
			
		||||
 | 
			
		||||
      - name: Build and release package
 | 
			
		||||
        run: |
 | 
			
		||||
          python3 -m pip install twine
 | 
			
		||||
          export TWINE_USERNAME="__token__"
 | 
			
		||||
          export TWINE_PASSWORD="${{ secrets.TWINE_TOKEN }}"
 | 
			
		||||
 | 
			
		||||
          script/release
 | 
			
		||||
 | 
			
		||||
  wheels-init:
 | 
			
		||||
    name: Init wheels build
 | 
			
		||||
    needs: release
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Generate requirements.txt
 | 
			
		||||
        run: |
 | 
			
		||||
          # Sleep to give pypi time to populate the new version across mirrors
 | 
			
		||||
          sleep 240
 | 
			
		||||
          version=$(echo "${{ github.ref }}" | awk -F"/" '{print $NF}' )
 | 
			
		||||
          echo "home-assistant-frontend==$version" > ./requirements.txt
 | 
			
		||||
 | 
			
		||||
      - name: Upload requirements.txt
 | 
			
		||||
        uses: actions/upload-artifact@v2
 | 
			
		||||
        with:
 | 
			
		||||
          name: requirements
 | 
			
		||||
          path: ./requirements.txt
 | 
			
		||||
 | 
			
		||||
  build-wheels:
 | 
			
		||||
    name: Build wheels for ${{ matrix.arch }}
 | 
			
		||||
    needs: wheels-init
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    strategy:
 | 
			
		||||
      matrix:
 | 
			
		||||
        arch: ["aarch64", "armhf", "armv7", "amd64", "i386"]
 | 
			
		||||
        tag:
 | 
			
		||||
          - "3.8-alpine3.12"
 | 
			
		||||
          - "3.9-alpine3.13"
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Download requirements.txt
 | 
			
		||||
        uses: actions/download-artifact@v2
 | 
			
		||||
        with:
 | 
			
		||||
          name: requirements
 | 
			
		||||
 | 
			
		||||
      - name: Build wheels
 | 
			
		||||
        uses: home-assistant/wheels@master
 | 
			
		||||
        with:
 | 
			
		||||
          tag: ${{ matrix.tag }}
 | 
			
		||||
          arch: ${{ matrix.arch }}
 | 
			
		||||
          wheels-host: ${{ secrets.WHEELS_HOST }}
 | 
			
		||||
          wheels-key: ${{ secrets.WHEELS_KEY }}
 | 
			
		||||
          wheels-user: wheels
 | 
			
		||||
          requirements: "requirements.txt"
 | 
			
		||||
							
								
								
									
										65
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,65 @@
 | 
			
		||||
name: Translations
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  schedule:
 | 
			
		||||
    - cron: "30 0 * * *"
 | 
			
		||||
  push:
 | 
			
		||||
    branches:
 | 
			
		||||
      - dev
 | 
			
		||||
    paths:
 | 
			
		||||
      - src/translations/en.json
 | 
			
		||||
 | 
			
		||||
env:
 | 
			
		||||
  NODE_VERSION: 12
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
  upload:
 | 
			
		||||
    name: Upload
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Checkout the repository
 | 
			
		||||
        uses: actions/checkout@v2
 | 
			
		||||
 | 
			
		||||
      - name: Set up Node ${{ env.NODE_VERSION }}
 | 
			
		||||
        uses: actions/setup-node@v2
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: ${{ env.NODE_VERSION }}
 | 
			
		||||
 | 
			
		||||
      - name: Upload Translations
 | 
			
		||||
        run: |
 | 
			
		||||
          export LOKALISE_TOKEN="${{ secrets.LOKALISE_TOKEN }}"
 | 
			
		||||
 | 
			
		||||
          ./script/translations_upload_base
 | 
			
		||||
 | 
			
		||||
  download:
 | 
			
		||||
    name: Download
 | 
			
		||||
    needs: upload
 | 
			
		||||
    if: github.event_name == 'schedule'
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Checkout the repository
 | 
			
		||||
        uses: actions/checkout@v2
 | 
			
		||||
 | 
			
		||||
      - name: Set up Node ${{ env.NODE_VERSION }}
 | 
			
		||||
        uses: actions/setup-node@v2
 | 
			
		||||
        with:
 | 
			
		||||
          node-version: ${{ env.NODE_VERSION }}
 | 
			
		||||
 | 
			
		||||
      - name: Download Translations
 | 
			
		||||
        run: |
 | 
			
		||||
          export LOKALISE_TOKEN="${{ secrets.LOKALISE_TOKEN }}"
 | 
			
		||||
 | 
			
		||||
          npm install
 | 
			
		||||
          ./script/translations_download
 | 
			
		||||
 | 
			
		||||
      - name: Initialize git
 | 
			
		||||
        uses: home-assistant/actions/helpers/git-init@master
 | 
			
		||||
        with:
 | 
			
		||||
          name: GitHub Action
 | 
			
		||||
          email: github-action@users.noreply.github.com
 | 
			
		||||
 | 
			
		||||
      - name: Update translation
 | 
			
		||||
        run: |
 | 
			
		||||
          git add translations
 | 
			
		||||
          git commit -am "Translation update"
 | 
			
		||||
          git push
 | 
			
		||||
							
								
								
									
										21
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										21
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -1,10 +1,17 @@
 | 
			
		||||
.DS_Store
 | 
			
		||||
.reify-cache
 | 
			
		||||
 | 
			
		||||
# build
 | 
			
		||||
build
 | 
			
		||||
build-translations/*
 | 
			
		||||
hass_frontend/*
 | 
			
		||||
dist
 | 
			
		||||
 | 
			
		||||
# yarn
 | 
			
		||||
.yarn
 | 
			
		||||
yarn-error.log
 | 
			
		||||
node_modules/*
 | 
			
		||||
npm-debug.log
 | 
			
		||||
.DS_Store
 | 
			
		||||
hass_frontend/*
 | 
			
		||||
.reify-cache
 | 
			
		||||
 | 
			
		||||
# Python stuff
 | 
			
		||||
*.py[cod]
 | 
			
		||||
@@ -14,11 +21,8 @@ hass_frontend/*
 | 
			
		||||
# venv stuff
 | 
			
		||||
pyvenv.cfg
 | 
			
		||||
pip-selfcheck.json
 | 
			
		||||
venv
 | 
			
		||||
venv/*
 | 
			
		||||
.venv
 | 
			
		||||
lib
 | 
			
		||||
bin
 | 
			
		||||
dist
 | 
			
		||||
 | 
			
		||||
# vscode
 | 
			
		||||
.vscode/*
 | 
			
		||||
@@ -31,9 +35,8 @@ src/cast/dev_const.ts
 | 
			
		||||
 | 
			
		||||
# Secrets
 | 
			
		||||
.lokalise_token
 | 
			
		||||
yarn-error.log
 | 
			
		||||
 | 
			
		||||
#asdf
 | 
			
		||||
# asdf
 | 
			
		||||
.tool-versions
 | 
			
		||||
 | 
			
		||||
# Home Assistant config
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +0,0 @@
 | 
			
		||||
jshint:
 | 
			
		||||
  enabled: false
 | 
			
		||||
 | 
			
		||||
eslint:
 | 
			
		||||
  enabled: true
 | 
			
		||||
  config_file: .eslintrc-hound.json
 | 
			
		||||
							
								
								
									
										4
									
								
								.mocharc.cjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.mocharc.cjs
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,4 @@
 | 
			
		||||
module.exports = {
 | 
			
		||||
  require: "test-mocha/testconf.js",
 | 
			
		||||
  timeout: 10000,
 | 
			
		||||
};
 | 
			
		||||
@@ -14,7 +14,7 @@ This is the repository for the official [Home Assistant](https://home-assistant.
 | 
			
		||||
- Development: [Instructions](https://developers.home-assistant.io/docs/frontend/development/)
 | 
			
		||||
- Production build: `script/build_frontend`
 | 
			
		||||
- Gallery: `cd gallery && script/develop_gallery`
 | 
			
		||||
- Hass.io: [Instructions](https://developers.home-assistant.io/docs/en/hassio_hass.html)
 | 
			
		||||
- Supervisor: [Instructions](https://developers.home-assistant.io/docs/supervisor/developing)
 | 
			
		||||
 | 
			
		||||
## Frontend development
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,30 +0,0 @@
 | 
			
		||||
# https://dev.azure.com/home-assistant
 | 
			
		||||
 | 
			
		||||
trigger: none
 | 
			
		||||
pr: none
 | 
			
		||||
schedules:
 | 
			
		||||
  - cron: "0 0 * * *"
 | 
			
		||||
    displayName: "build preview"
 | 
			
		||||
    branches:
 | 
			
		||||
      include:
 | 
			
		||||
      - dev
 | 
			
		||||
    always: true
 | 
			
		||||
variables:
 | 
			
		||||
  - group: netlify
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
 | 
			
		||||
- job: 'Netlify_preview'
 | 
			
		||||
  pool:
 | 
			
		||||
    vmImage: 'ubuntu-latest'
 | 
			
		||||
  steps:
 | 
			
		||||
  - script: |
 | 
			
		||||
      # Cast
 | 
			
		||||
      curl -X POST -d {} https://api.netlify.com/build_hooks/${NETLIFY_CAST}
 | 
			
		||||
 | 
			
		||||
      # Demo
 | 
			
		||||
      curl -X POST -d {} https://api.netlify.com/build_hooks/${NETLIFY_DEMO}
 | 
			
		||||
 | 
			
		||||
      # Gallery
 | 
			
		||||
      curl -X POST -d {} https://api.netlify.com/build_hooks/${NETLIFY_GALLERY}
 | 
			
		||||
    displayName: 'Trigger netlify build preview'
 | 
			
		||||
@@ -1,59 +0,0 @@
 | 
			
		||||
# https://dev.azure.com/home-assistant
 | 
			
		||||
 | 
			
		||||
trigger:
 | 
			
		||||
  batch: true
 | 
			
		||||
  tags:
 | 
			
		||||
    include:
 | 
			
		||||
      - "*"
 | 
			
		||||
pr: none
 | 
			
		||||
variables:
 | 
			
		||||
  - name: versionWheels
 | 
			
		||||
    value: '1.10.1-3.7-alpine3.11'
 | 
			
		||||
  - name: versionNode
 | 
			
		||||
    value: '12.1'
 | 
			
		||||
  - group: twine
 | 
			
		||||
resources:
 | 
			
		||||
  repositories:
 | 
			
		||||
    - repository: azure
 | 
			
		||||
      type: github
 | 
			
		||||
      name: 'home-assistant/ci-azure'
 | 
			
		||||
      endpoint: 'home-assistant'
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
stages:
 | 
			
		||||
  - stage: "Validate"
 | 
			
		||||
    jobs:
 | 
			
		||||
    - template: templates/azp-job-version.yaml@azure
 | 
			
		||||
 | 
			
		||||
  - stage: "Build"
 | 
			
		||||
    jobs:
 | 
			
		||||
      - job: "ReleasePython"
 | 
			
		||||
        pool:
 | 
			
		||||
          vmImage: "ubuntu-latest"
 | 
			
		||||
        steps:
 | 
			
		||||
          - task: UsePythonVersion@0
 | 
			
		||||
            displayName: "Use Python 3.7"
 | 
			
		||||
            inputs:
 | 
			
		||||
              versionSpec: "3.7"
 | 
			
		||||
          - task: NodeTool@0
 | 
			
		||||
            displayName: "Use Node $(versionNode)"
 | 
			
		||||
            inputs:
 | 
			
		||||
              versionSpec: "$(versionNode)"
 | 
			
		||||
          - script: pip install twine wheel
 | 
			
		||||
            displayName: "Install tools"
 | 
			
		||||
          - script: |
 | 
			
		||||
              export TWINE_USERNAME="$(twineUser)"
 | 
			
		||||
              export TWINE_PASSWORD="$(twinePassword)"
 | 
			
		||||
 | 
			
		||||
              script/release
 | 
			
		||||
            displayName: "Build and release package"
 | 
			
		||||
  - stage: "Wheels"
 | 
			
		||||
    jobs:
 | 
			
		||||
      - template: templates/azp-job-wheels.yaml@azure
 | 
			
		||||
        parameters:
 | 
			
		||||
          builderVersion: '$(versionWheels)'
 | 
			
		||||
          wheelsRequirement: 'requirement.txt'
 | 
			
		||||
          preBuild:
 | 
			
		||||
          - script: |
 | 
			
		||||
              sleep 240
 | 
			
		||||
              echo "home-assistant-frontend==$(Build.SourceBranchName)" > requirement.txt
 | 
			
		||||
@@ -1,70 +0,0 @@
 | 
			
		||||
# https://dev.azure.com/home-assistant
 | 
			
		||||
 | 
			
		||||
trigger:
 | 
			
		||||
  batch: true
 | 
			
		||||
  branches:
 | 
			
		||||
    include:
 | 
			
		||||
    - dev
 | 
			
		||||
  paths:
 | 
			
		||||
    include:
 | 
			
		||||
    - translations/en.json
 | 
			
		||||
pr: none
 | 
			
		||||
schedules:
 | 
			
		||||
  - cron: "30 0 * * *"
 | 
			
		||||
    displayName: "frontend translation update"
 | 
			
		||||
    branches:
 | 
			
		||||
      include:
 | 
			
		||||
      - dev
 | 
			
		||||
    always: true
 | 
			
		||||
variables:
 | 
			
		||||
- group: translation
 | 
			
		||||
resources:
 | 
			
		||||
  repositories:
 | 
			
		||||
  - repository: azure
 | 
			
		||||
    type: github
 | 
			
		||||
    name: 'home-assistant/ci-azure'
 | 
			
		||||
    endpoint: 'home-assistant'
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
jobs:
 | 
			
		||||
 | 
			
		||||
- job: 'Upload'
 | 
			
		||||
  pool:
 | 
			
		||||
    vmImage: 'ubuntu-latest'
 | 
			
		||||
  steps:
 | 
			
		||||
  - task: NodeTool@0
 | 
			
		||||
    displayName: 'Use Node 12.x'
 | 
			
		||||
    inputs:
 | 
			
		||||
      versionSpec: '12.x'
 | 
			
		||||
  - script: |
 | 
			
		||||
      export LOKALISE_TOKEN="$(lokaliseToken)"
 | 
			
		||||
      export AZURE_BRANCH="$(Build.SourceBranchName)"
 | 
			
		||||
 | 
			
		||||
      ./script/translations_upload_base
 | 
			
		||||
    displayName: 'Upload Translation'
 | 
			
		||||
 | 
			
		||||
- job: 'Download'
 | 
			
		||||
  dependsOn:
 | 
			
		||||
  - 'Upload'
 | 
			
		||||
  condition: or(eq(variables['Build.Reason'], 'Schedule'), eq(variables['Build.Reason'], 'Manual'))
 | 
			
		||||
  pool:
 | 
			
		||||
    vmImage: 'ubuntu-latest'
 | 
			
		||||
  steps:
 | 
			
		||||
  - task: NodeTool@0
 | 
			
		||||
    displayName: 'Use Node 12.x'
 | 
			
		||||
    inputs:
 | 
			
		||||
      versionSpec: '12.x'
 | 
			
		||||
  - template: templates/azp-step-git-init.yaml@azure
 | 
			
		||||
  - script: |
 | 
			
		||||
      export LOKALISE_TOKEN="$(lokaliseToken)"
 | 
			
		||||
      export AZURE_BRANCH="$(Build.SourceBranchName)"
 | 
			
		||||
 | 
			
		||||
      npm install
 | 
			
		||||
      ./script/translations_download
 | 
			
		||||
    displayName: 'Download Translation'
 | 
			
		||||
  - script: |
 | 
			
		||||
      git checkout dev
 | 
			
		||||
      git add translation
 | 
			
		||||
      git commit -am "[ci skip] Translation update"
 | 
			
		||||
      git push
 | 
			
		||||
    displayName: 'Update translation'
 | 
			
		||||
							
								
								
									
										39
									
								
								build-scripts/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								build-scripts/README.md
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,39 @@
 | 
			
		||||
# Bundling Home Assistant Frontend
 | 
			
		||||
 | 
			
		||||
The Home Assistant build pipeline contains various steps to prepare a build.
 | 
			
		||||
 | 
			
		||||
- Generating icon files to be included
 | 
			
		||||
- Generating translation files to be included
 | 
			
		||||
- Converting TypeScript, CSS and JSON files to JavaScript
 | 
			
		||||
- Bundling
 | 
			
		||||
- Minifying the files
 | 
			
		||||
- Generating the HTML entrypoint files
 | 
			
		||||
- Generating the service worker
 | 
			
		||||
- Compressing the files
 | 
			
		||||
 | 
			
		||||
## Converting files
 | 
			
		||||
 | 
			
		||||
Currently in Home Assistant we use a bundler to convert TypeScript, CSS and JSON files to JavaScript files that the browser understands.
 | 
			
		||||
 | 
			
		||||
We currently rely on Webpack but also have experimental Rollup support. Both of these programs bundle the converted files in both production and development.
 | 
			
		||||
 | 
			
		||||
For development, bundling is optional. We just want to get the right files in the browser.
 | 
			
		||||
 | 
			
		||||
Responsibilities of the converter during development:
 | 
			
		||||
 | 
			
		||||
- Convert TypeScript to JavaScript
 | 
			
		||||
- Convert CSS to JavaScript that sets the content as the default export
 | 
			
		||||
- Convert JSON to JavaScript that sets the content as the default export
 | 
			
		||||
- Make sure import, dynamic import and web worker references work
 | 
			
		||||
  - Add extensions where missing
 | 
			
		||||
  - Resolve absolute package imports
 | 
			
		||||
- Filter out specific imports/packages
 | 
			
		||||
- Replace constants with values
 | 
			
		||||
 | 
			
		||||
In production, the following responsibilities are added:
 | 
			
		||||
 | 
			
		||||
- Minify HTML
 | 
			
		||||
- Bundle multiple imports so that the browser can fetch less files
 | 
			
		||||
- Generate a second version that is ES5 compatible
 | 
			
		||||
 | 
			
		||||
Configuration for all these steps are specified in [bundle.js](bundle.js).
 | 
			
		||||
@@ -1,3 +1,4 @@
 | 
			
		||||
/* eslint-disable @typescript-eslint/no-var-requires */
 | 
			
		||||
const path = require("path");
 | 
			
		||||
const env = require("./env.js");
 | 
			
		||||
const paths = require("./paths.js");
 | 
			
		||||
@@ -44,22 +45,23 @@ module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
module.exports.terserOptions = (latestBuild) => ({
 | 
			
		||||
  safari10: true,
 | 
			
		||||
  safari10: !latestBuild,
 | 
			
		||||
  ecma: latestBuild ? undefined : 5,
 | 
			
		||||
  output: { comments: false },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
module.exports.babelOptions = ({ latestBuild }) => ({
 | 
			
		||||
  babelrc: false,
 | 
			
		||||
  compact: false,
 | 
			
		||||
  presets: [
 | 
			
		||||
    !latestBuild && [
 | 
			
		||||
      require("@babel/preset-env").default,
 | 
			
		||||
      "@babel/preset-env",
 | 
			
		||||
      {
 | 
			
		||||
        useBuiltIns: "entry",
 | 
			
		||||
        corejs: "3.6",
 | 
			
		||||
      },
 | 
			
		||||
    ],
 | 
			
		||||
    require("@babel/preset-typescript").default,
 | 
			
		||||
    "@babel/preset-typescript",
 | 
			
		||||
  ].filter(Boolean),
 | 
			
		||||
  plugins: [
 | 
			
		||||
    // Part of ES2018. Converts {...a, b: 2} to Object.assign({}, a, {b: 2})
 | 
			
		||||
@@ -72,23 +74,12 @@ module.exports.babelOptions = ({ latestBuild }) => ({
 | 
			
		||||
    "@babel/plugin-syntax-dynamic-import",
 | 
			
		||||
    "@babel/plugin-proposal-optional-chaining",
 | 
			
		||||
    "@babel/plugin-proposal-nullish-coalescing-operator",
 | 
			
		||||
    [
 | 
			
		||||
      require("@babel/plugin-proposal-decorators").default,
 | 
			
		||||
      { decoratorsBeforeExport: true },
 | 
			
		||||
    ],
 | 
			
		||||
    [
 | 
			
		||||
      require("@babel/plugin-proposal-class-properties").default,
 | 
			
		||||
      { loose: true },
 | 
			
		||||
    ],
 | 
			
		||||
    ["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }],
 | 
			
		||||
    ["@babel/plugin-proposal-private-methods", { loose: true }],
 | 
			
		||||
    ["@babel/plugin-proposal-class-properties", { loose: true }],
 | 
			
		||||
  ].filter(Boolean),
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
// Are already ES5, cause warnings when babelified.
 | 
			
		||||
module.exports.babelExclude = () => [
 | 
			
		||||
  require.resolve("@mdi/js/mdi.js"),
 | 
			
		||||
  require.resolve("hls.js"),
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const outputPath = (outputRoot, latestBuild) =>
 | 
			
		||||
  path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5");
 | 
			
		||||
 | 
			
		||||
@@ -117,7 +108,7 @@ BundleConfig {
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
module.exports.config = {
 | 
			
		||||
  app({ isProdBuild, latestBuild, isStatsBuild }) {
 | 
			
		||||
  app({ isProdBuild, latestBuild, isStatsBuild, isWDS }) {
 | 
			
		||||
    return {
 | 
			
		||||
      entry: {
 | 
			
		||||
        service_worker: "./src/entrypoints/service_worker.ts",
 | 
			
		||||
@@ -132,6 +123,7 @@ module.exports.config = {
 | 
			
		||||
      isProdBuild,
 | 
			
		||||
      latestBuild,
 | 
			
		||||
      isStatsBuild,
 | 
			
		||||
      isWDS,
 | 
			
		||||
    };
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,4 @@
 | 
			
		||||
/* eslint-disable @typescript-eslint/no-var-requires */
 | 
			
		||||
const fs = require("fs");
 | 
			
		||||
const path = require("path");
 | 
			
		||||
const paths = require("./paths.js");
 | 
			
		||||
@@ -6,6 +7,9 @@ module.exports = {
 | 
			
		||||
  useRollup() {
 | 
			
		||||
    return process.env.ROLLUP === "1";
 | 
			
		||||
  },
 | 
			
		||||
  useWDS() {
 | 
			
		||||
    return process.env.WDS === "1";
 | 
			
		||||
  },
 | 
			
		||||
  isProdBuild() {
 | 
			
		||||
    return (
 | 
			
		||||
      process.env.NODE_ENV === "production" || module.exports.isStatsBuild()
 | 
			
		||||
 
 | 
			
		||||
@@ -12,6 +12,7 @@ require("./webpack.js");
 | 
			
		||||
require("./service-worker.js");
 | 
			
		||||
require("./entry-html.js");
 | 
			
		||||
require("./rollup.js");
 | 
			
		||||
require("./wds.js");
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "develop-app",
 | 
			
		||||
@@ -28,7 +29,11 @@ gulp.task(
 | 
			
		||||
      "build-translations"
 | 
			
		||||
    ),
 | 
			
		||||
    "copy-static-app",
 | 
			
		||||
    env.useRollup() ? "rollup-watch-app" : "webpack-watch-app"
 | 
			
		||||
    env.useWDS()
 | 
			
		||||
      ? "wds-watch-app"
 | 
			
		||||
      : env.useRollup()
 | 
			
		||||
      ? "rollup-watch-app"
 | 
			
		||||
      : "webpack-watch-app"
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -19,6 +19,7 @@ const renderTemplate = (pth, data = {}, pathFunc = templatePath) => {
 | 
			
		||||
  return compiled({
 | 
			
		||||
    ...data,
 | 
			
		||||
    useRollup: env.useRollup(),
 | 
			
		||||
    useWDS: env.useWDS(),
 | 
			
		||||
    renderTemplate,
 | 
			
		||||
  });
 | 
			
		||||
};
 | 
			
		||||
@@ -90,10 +91,23 @@ gulp.task("gen-pages-prod", (done) => {
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task("gen-index-app-dev", (done) => {
 | 
			
		||||
  let latestAppJS, latestCoreJS, latestCustomPanelJS;
 | 
			
		||||
 | 
			
		||||
  if (env.useWDS()) {
 | 
			
		||||
    latestAppJS = "http://localhost:8000/src/entrypoints/app.ts";
 | 
			
		||||
    latestCoreJS = "http://localhost:8000/src/entrypoints/core.ts";
 | 
			
		||||
    latestCustomPanelJS =
 | 
			
		||||
      "http://localhost:8000/src/entrypoints/custom-panel.ts";
 | 
			
		||||
  } else {
 | 
			
		||||
    latestAppJS = "/frontend_latest/app.js";
 | 
			
		||||
    latestCoreJS = "/frontend_latest/core.js";
 | 
			
		||||
    latestCustomPanelJS = "/frontend_latest/custom-panel.js";
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  const content = renderTemplate("index", {
 | 
			
		||||
    latestAppJS: "/frontend_latest/app.js",
 | 
			
		||||
    latestCoreJS: "/frontend_latest/core.js",
 | 
			
		||||
    latestCustomPanelJS: "/frontend_latest/custom-panel.js",
 | 
			
		||||
    latestAppJS,
 | 
			
		||||
    latestCoreJS,
 | 
			
		||||
    latestCustomPanelJS,
 | 
			
		||||
 | 
			
		||||
    es5AppJS: "/frontend_es5/app.js",
 | 
			
		||||
    es5CoreJS: "/frontend_es5/core.js",
 | 
			
		||||
 
 | 
			
		||||
@@ -85,6 +85,11 @@ gulp.task("copy-translations-app", async () => {
 | 
			
		||||
  copyTranslations(staticDir);
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task("copy-translations-supervisor", async () => {
 | 
			
		||||
  const staticDir = paths.hassio_output_static;
 | 
			
		||||
  copyTranslations(staticDir);
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task("copy-static-app", async () => {
 | 
			
		||||
  const staticDir = paths.app_output_static;
 | 
			
		||||
  // Basic static files
 | 
			
		||||
 
 | 
			
		||||
@@ -10,6 +10,8 @@ require("./gen-icons-json.js");
 | 
			
		||||
require("./webpack.js");
 | 
			
		||||
require("./compress.js");
 | 
			
		||||
require("./rollup.js");
 | 
			
		||||
require("./gather-static.js");
 | 
			
		||||
require("./translations.js");
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "develop-hassio",
 | 
			
		||||
@@ -20,6 +22,8 @@ gulp.task(
 | 
			
		||||
    "clean-hassio",
 | 
			
		||||
    "gen-icons-json",
 | 
			
		||||
    "gen-index-hassio-dev",
 | 
			
		||||
    "build-supervisor-translations",
 | 
			
		||||
    "copy-translations-supervisor",
 | 
			
		||||
    env.useRollup() ? "rollup-watch-hassio" : "webpack-watch-hassio"
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
@@ -32,6 +36,8 @@ gulp.task(
 | 
			
		||||
    },
 | 
			
		||||
    "clean-hassio",
 | 
			
		||||
    "gen-icons-json",
 | 
			
		||||
    "build-supervisor-translations",
 | 
			
		||||
    "copy-translations-supervisor",
 | 
			
		||||
    env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio",
 | 
			
		||||
    "gen-index-hassio-prod",
 | 
			
		||||
    ...// Don't compress running tests
 | 
			
		||||
 
 | 
			
		||||
@@ -33,21 +33,10 @@ String.prototype.rsplit = function (sep, maxsplit) {
 | 
			
		||||
    : split;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Panel translations which should be split from the core translations. These
 | 
			
		||||
// should mirror the fragment definitions in polymer.json, so that we load
 | 
			
		||||
// additional resources at equivalent points.
 | 
			
		||||
const TRANSLATION_FRAGMENTS = [
 | 
			
		||||
  "config",
 | 
			
		||||
  "history",
 | 
			
		||||
  "logbook",
 | 
			
		||||
  "mailbox",
 | 
			
		||||
  "profile",
 | 
			
		||||
  "shopping-list",
 | 
			
		||||
  "page-authorize",
 | 
			
		||||
  "page-demo",
 | 
			
		||||
  "page-onboarding",
 | 
			
		||||
  "developer-tools",
 | 
			
		||||
];
 | 
			
		||||
// Panel translations which should be split from the core translations.
 | 
			
		||||
const TRANSLATION_FRAGMENTS = Object.keys(
 | 
			
		||||
  require("../../src/translations/en.json").ui.panel
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
function recursiveFlatten(prefix, data) {
 | 
			
		||||
  let output = {};
 | 
			
		||||
@@ -277,6 +266,7 @@ gulp.task(taskName, function () {
 | 
			
		||||
        TRANSLATION_FRAGMENTS.forEach((fragment) => {
 | 
			
		||||
          delete data.ui.panel[fragment];
 | 
			
		||||
        });
 | 
			
		||||
        delete data.supervisor;
 | 
			
		||||
        return data;
 | 
			
		||||
      })
 | 
			
		||||
    )
 | 
			
		||||
@@ -353,6 +343,62 @@ gulp.task(
 | 
			
		||||
  }
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
gulp.task("build-translation-fragment-supervisor", function () {
 | 
			
		||||
  return gulp
 | 
			
		||||
    .src(fullDir + "/*.json")
 | 
			
		||||
    .pipe(transform((data) => data.supervisor))
 | 
			
		||||
    .pipe(gulp.dest(workDir + "/supervisor"));
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task("build-translation-flatten-supervisor", function () {
 | 
			
		||||
  return gulp
 | 
			
		||||
    .src(workDir + "/supervisor/*.json")
 | 
			
		||||
    .pipe(
 | 
			
		||||
      transform(function (data) {
 | 
			
		||||
        // Polymer.AppLocalizeBehavior requires flattened json
 | 
			
		||||
        return flatten(data);
 | 
			
		||||
      })
 | 
			
		||||
    )
 | 
			
		||||
    .pipe(gulp.dest(outDir));
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task("build-translation-write-metadata", function writeMetadata() {
 | 
			
		||||
  return gulp
 | 
			
		||||
    .src(
 | 
			
		||||
      [
 | 
			
		||||
        path.join(paths.translations_src, "translationMetadata.json"),
 | 
			
		||||
        workDir + "/testMetadata.json",
 | 
			
		||||
        workDir + "/translationFingerprints.json",
 | 
			
		||||
      ],
 | 
			
		||||
      { allowEmpty: true }
 | 
			
		||||
    )
 | 
			
		||||
    .pipe(merge({}))
 | 
			
		||||
    .pipe(
 | 
			
		||||
      transform(function (data) {
 | 
			
		||||
        const newData = {};
 | 
			
		||||
        Object.entries(data).forEach(([key, value]) => {
 | 
			
		||||
          // Filter out translations without native name.
 | 
			
		||||
          if (value.nativeName) {
 | 
			
		||||
            newData[key] = value;
 | 
			
		||||
          } else {
 | 
			
		||||
            console.warn(
 | 
			
		||||
              `Skipping language ${key}. Native name was not translated.`
 | 
			
		||||
            );
 | 
			
		||||
          }
 | 
			
		||||
        });
 | 
			
		||||
        return newData;
 | 
			
		||||
      })
 | 
			
		||||
    )
 | 
			
		||||
    .pipe(
 | 
			
		||||
      transform((data) => ({
 | 
			
		||||
        fragments: TRANSLATION_FRAGMENTS,
 | 
			
		||||
        translations: data,
 | 
			
		||||
      }))
 | 
			
		||||
    )
 | 
			
		||||
    .pipe(rename("translationMetadata.json"))
 | 
			
		||||
    .pipe(gulp.dest(workDir));
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "build-translations",
 | 
			
		||||
  gulp.series(
 | 
			
		||||
@@ -364,42 +410,20 @@ gulp.task(
 | 
			
		||||
    gulp.parallel(...splitTasks),
 | 
			
		||||
    "build-flattened-translations",
 | 
			
		||||
    "build-translation-fingerprints",
 | 
			
		||||
    function writeMetadata() {
 | 
			
		||||
      return gulp
 | 
			
		||||
        .src(
 | 
			
		||||
          [
 | 
			
		||||
            path.join(paths.translations_src, "translationMetadata.json"),
 | 
			
		||||
            workDir + "/testMetadata.json",
 | 
			
		||||
            workDir + "/translationFingerprints.json",
 | 
			
		||||
          ],
 | 
			
		||||
          { allowEmpty: true }
 | 
			
		||||
        )
 | 
			
		||||
        .pipe(merge({}))
 | 
			
		||||
        .pipe(
 | 
			
		||||
          transform(function (data) {
 | 
			
		||||
            const newData = {};
 | 
			
		||||
            Object.entries(data).forEach(([key, value]) => {
 | 
			
		||||
              // Filter out translations without native name.
 | 
			
		||||
              if (data[key].nativeName) {
 | 
			
		||||
                newData[key] = data[key];
 | 
			
		||||
              } else {
 | 
			
		||||
                console.warn(
 | 
			
		||||
                  `Skipping language ${key}. Native name was not translated.`
 | 
			
		||||
                );
 | 
			
		||||
              }
 | 
			
		||||
              if (data[key]) newData[key] = value;
 | 
			
		||||
            });
 | 
			
		||||
            return newData;
 | 
			
		||||
          })
 | 
			
		||||
        )
 | 
			
		||||
        .pipe(
 | 
			
		||||
          transform((data) => ({
 | 
			
		||||
            fragments: TRANSLATION_FRAGMENTS,
 | 
			
		||||
            translations: data,
 | 
			
		||||
          }))
 | 
			
		||||
        )
 | 
			
		||||
        .pipe(rename("translationMetadata.json"))
 | 
			
		||||
        .pipe(gulp.dest(workDir));
 | 
			
		||||
    }
 | 
			
		||||
    "build-translation-write-metadata"
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "build-supervisor-translations",
 | 
			
		||||
  gulp.series(
 | 
			
		||||
    "clean-translations",
 | 
			
		||||
    "ensure-translations-build-dir",
 | 
			
		||||
    "build-master-translation",
 | 
			
		||||
    "build-merged-translations",
 | 
			
		||||
    "build-translation-fragment-supervisor",
 | 
			
		||||
    "build-translation-flatten-supervisor",
 | 
			
		||||
    "build-translation-fingerprints",
 | 
			
		||||
    "build-translation-write-metadata"
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										11
									
								
								build-scripts/gulp/wds.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								build-scripts/gulp/wds.js
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,11 @@
 | 
			
		||||
// Tasks to run Rollup
 | 
			
		||||
const gulp = require("gulp");
 | 
			
		||||
const { startDevServer } = require("@web/dev-server");
 | 
			
		||||
 | 
			
		||||
gulp.task("wds-watch-app", () => {
 | 
			
		||||
  startDevServer({
 | 
			
		||||
    config: {
 | 
			
		||||
      watch: true,
 | 
			
		||||
    },
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
@@ -1,4 +1,5 @@
 | 
			
		||||
// Tasks to run webpack.
 | 
			
		||||
const fs = require("fs");
 | 
			
		||||
const gulp = require("gulp");
 | 
			
		||||
const webpack = require("webpack");
 | 
			
		||||
const WebpackDevServer = require("webpack-dev-server");
 | 
			
		||||
@@ -18,6 +19,11 @@ const bothBuilds = (createConfigFunc, params) => [
 | 
			
		||||
  createConfigFunc({ ...params, latestBuild: false }),
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const isWsl = fs
 | 
			
		||||
  .readFileSync("/proc/version", "utf-8")
 | 
			
		||||
  .toLocaleLowerCase()
 | 
			
		||||
  .includes("microsoft");
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * @param {{
 | 
			
		||||
 *   compiler: import("webpack").Compiler,
 | 
			
		||||
@@ -47,7 +53,7 @@ const runDevServer = ({
 | 
			
		||||
    );
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
const handler = (done) => (err, stats) => {
 | 
			
		||||
const doneHandler = (done) => (err, stats) => {
 | 
			
		||||
  if (err) {
 | 
			
		||||
    log.error(err.stack || err);
 | 
			
		||||
    if (err.details) {
 | 
			
		||||
@@ -67,11 +73,20 @@ const handler = (done) => (err, stats) => {
 | 
			
		||||
  }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const prodBuild = (conf) =>
 | 
			
		||||
  new Promise((resolve) => {
 | 
			
		||||
    webpack(
 | 
			
		||||
      conf,
 | 
			
		||||
      // Resolve promise when done. Because we pass a callback, webpack closes itself
 | 
			
		||||
      doneHandler(resolve)
 | 
			
		||||
    );
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
gulp.task("webpack-watch-app", () => {
 | 
			
		||||
  // we are not calling done, so this command will run forever
 | 
			
		||||
  // This command will run forever because we don't close compiler
 | 
			
		||||
  webpack(createAppConfig({ isProdBuild: false, latestBuild: true })).watch(
 | 
			
		||||
    { ignored: /build-translations/ },
 | 
			
		||||
    handler()
 | 
			
		||||
    { ignored: /build-translations/, poll: isWsl },
 | 
			
		||||
    doneHandler()
 | 
			
		||||
  );
 | 
			
		||||
  gulp.watch(
 | 
			
		||||
    path.join(paths.translations_src, "en.json"),
 | 
			
		||||
@@ -79,15 +94,12 @@ gulp.task("webpack-watch-app", () => {
 | 
			
		||||
  );
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "webpack-prod-app",
 | 
			
		||||
  () =>
 | 
			
		||||
    new Promise((resolve) =>
 | 
			
		||||
      webpack(
 | 
			
		||||
        bothBuilds(createAppConfig, { isProdBuild: true }),
 | 
			
		||||
        handler(resolve)
 | 
			
		||||
      )
 | 
			
		||||
    )
 | 
			
		||||
gulp.task("webpack-prod-app", () =>
 | 
			
		||||
  prodBuild(
 | 
			
		||||
    bothBuilds(createAppConfig, {
 | 
			
		||||
      isProdBuild: true,
 | 
			
		||||
    })
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
gulp.task("webpack-dev-server-demo", () => {
 | 
			
		||||
@@ -98,17 +110,12 @@ gulp.task("webpack-dev-server-demo", () => {
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "webpack-prod-demo",
 | 
			
		||||
  () =>
 | 
			
		||||
    new Promise((resolve) =>
 | 
			
		||||
      webpack(
 | 
			
		||||
        bothBuilds(createDemoConfig, {
 | 
			
		||||
          isProdBuild: true,
 | 
			
		||||
        }),
 | 
			
		||||
        handler(resolve)
 | 
			
		||||
      )
 | 
			
		||||
    )
 | 
			
		||||
gulp.task("webpack-prod-demo", () =>
 | 
			
		||||
  prodBuild(
 | 
			
		||||
    bothBuilds(createDemoConfig, {
 | 
			
		||||
      isProdBuild: true,
 | 
			
		||||
    })
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
gulp.task("webpack-dev-server-cast", () => {
 | 
			
		||||
@@ -121,41 +128,35 @@ gulp.task("webpack-dev-server-cast", () => {
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "webpack-prod-cast",
 | 
			
		||||
  () =>
 | 
			
		||||
    new Promise((resolve) =>
 | 
			
		||||
      webpack(
 | 
			
		||||
        bothBuilds(createCastConfig, {
 | 
			
		||||
          isProdBuild: true,
 | 
			
		||||
        }),
 | 
			
		||||
 | 
			
		||||
        handler(resolve)
 | 
			
		||||
      )
 | 
			
		||||
    )
 | 
			
		||||
gulp.task("webpack-prod-cast", () =>
 | 
			
		||||
  prodBuild(
 | 
			
		||||
    bothBuilds(createCastConfig, {
 | 
			
		||||
      isProdBuild: true,
 | 
			
		||||
    })
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
gulp.task("webpack-watch-hassio", () => {
 | 
			
		||||
  // we are not calling done, so this command will run forever
 | 
			
		||||
  // This command will run forever because we don't close compiler
 | 
			
		||||
  webpack(
 | 
			
		||||
    createHassioConfig({
 | 
			
		||||
      isProdBuild: false,
 | 
			
		||||
      latestBuild: true,
 | 
			
		||||
    })
 | 
			
		||||
  ).watch({}, handler());
 | 
			
		||||
  ).watch({ ignored: /build-translations/, poll: isWsl }, doneHandler());
 | 
			
		||||
 | 
			
		||||
  gulp.watch(
 | 
			
		||||
    path.join(paths.translations_src, "en.json"),
 | 
			
		||||
    gulp.series("build-supervisor-translations", "copy-translations-supervisor")
 | 
			
		||||
  );
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "webpack-prod-hassio",
 | 
			
		||||
  () =>
 | 
			
		||||
    new Promise((resolve) =>
 | 
			
		||||
      webpack(
 | 
			
		||||
        bothBuilds(createHassioConfig, {
 | 
			
		||||
          isProdBuild: true,
 | 
			
		||||
        }),
 | 
			
		||||
        handler(resolve)
 | 
			
		||||
      )
 | 
			
		||||
    )
 | 
			
		||||
gulp.task("webpack-prod-hassio", () =>
 | 
			
		||||
  prodBuild(
 | 
			
		||||
    bothBuilds(createHassioConfig, {
 | 
			
		||||
      isProdBuild: true,
 | 
			
		||||
    })
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
gulp.task("webpack-dev-server-gallery", () => {
 | 
			
		||||
@@ -167,17 +168,11 @@ gulp.task("webpack-dev-server-gallery", () => {
 | 
			
		||||
  });
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
gulp.task(
 | 
			
		||||
  "webpack-prod-gallery",
 | 
			
		||||
  () =>
 | 
			
		||||
    new Promise((resolve) =>
 | 
			
		||||
      webpack(
 | 
			
		||||
        createGalleryConfig({
 | 
			
		||||
          isProdBuild: true,
 | 
			
		||||
          latestBuild: true,
 | 
			
		||||
        }),
 | 
			
		||||
 | 
			
		||||
        handler(resolve)
 | 
			
		||||
      )
 | 
			
		||||
    )
 | 
			
		||||
gulp.task("webpack-prod-gallery", () =>
 | 
			
		||||
  prodBuild(
 | 
			
		||||
    createGalleryConfig({
 | 
			
		||||
      isProdBuild: true,
 | 
			
		||||
      latestBuild: true,
 | 
			
		||||
    })
 | 
			
		||||
  )
 | 
			
		||||
);
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,4 @@
 | 
			
		||||
/* eslint-disable @typescript-eslint/no-var-requires */
 | 
			
		||||
const path = require("path");
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
@@ -34,6 +35,7 @@ module.exports = {
 | 
			
		||||
 | 
			
		||||
  hassio_dir: path.resolve(__dirname, "../hassio"),
 | 
			
		||||
  hassio_output_root: path.resolve(__dirname, "../hassio/build"),
 | 
			
		||||
  hassio_output_static: path.resolve(__dirname, "../hassio/build/static"),
 | 
			
		||||
  hassio_output_latest: path.resolve(
 | 
			
		||||
    __dirname,
 | 
			
		||||
    "../hassio/build/frontend_latest"
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,3 @@
 | 
			
		||||
const path = require("path");
 | 
			
		||||
 | 
			
		||||
module.exports = function (userOptions = {}) {
 | 
			
		||||
  // Files need to be absolute paths.
 | 
			
		||||
  // This only works if the file has no exports
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,10 @@
 | 
			
		||||
/* eslint-disable @typescript-eslint/no-var-requires */
 | 
			
		||||
const path = require("path");
 | 
			
		||||
 | 
			
		||||
const commonjs = require("@rollup/plugin-commonjs");
 | 
			
		||||
const resolve = require("@rollup/plugin-node-resolve");
 | 
			
		||||
const json = require("@rollup/plugin-json");
 | 
			
		||||
const babel = require("rollup-plugin-babel");
 | 
			
		||||
const babel = require("@rollup/plugin-babel").babel;
 | 
			
		||||
const replace = require("@rollup/plugin-replace");
 | 
			
		||||
const visualizer = require("rollup-plugin-visualizer");
 | 
			
		||||
const { string } = require("rollup-plugin-string");
 | 
			
		||||
@@ -31,116 +32,103 @@ const createRollupConfig = ({
 | 
			
		||||
  isStatsBuild,
 | 
			
		||||
  publicPath,
 | 
			
		||||
  dontHash,
 | 
			
		||||
}) => {
 | 
			
		||||
  return {
 | 
			
		||||
    /**
 | 
			
		||||
     * @type { import("rollup").InputOptions }
 | 
			
		||||
     */
 | 
			
		||||
    inputOptions: {
 | 
			
		||||
      input: entry,
 | 
			
		||||
      // Some entry points contain no JavaScript. This setting silences a warning about that.
 | 
			
		||||
      // https://rollupjs.org/guide/en/#preserveentrysignatures
 | 
			
		||||
      preserveEntrySignatures: false,
 | 
			
		||||
      plugins: [
 | 
			
		||||
        ignore({
 | 
			
		||||
          files: bundle.emptyPackages({ latestBuild }),
 | 
			
		||||
        }),
 | 
			
		||||
        resolve({
 | 
			
		||||
          extensions,
 | 
			
		||||
          preferBuiltins: false,
 | 
			
		||||
          browser: true,
 | 
			
		||||
          rootDir: paths.polymer_dir,
 | 
			
		||||
        }),
 | 
			
		||||
        commonjs({
 | 
			
		||||
          namedExports: {
 | 
			
		||||
            "js-yaml": ["safeDump", "safeLoad"],
 | 
			
		||||
          },
 | 
			
		||||
        }),
 | 
			
		||||
        json(),
 | 
			
		||||
        babel({
 | 
			
		||||
          ...bundle.babelOptions({ latestBuild }),
 | 
			
		||||
          extensions,
 | 
			
		||||
          exclude: bundle.babelExclude(),
 | 
			
		||||
        }),
 | 
			
		||||
        string({
 | 
			
		||||
          // Import certain extensions as strings
 | 
			
		||||
          include: [path.join(paths.polymer_dir, "node_modules/**/*.css")],
 | 
			
		||||
        }),
 | 
			
		||||
        replace(
 | 
			
		||||
          bundle.definedVars({ isProdBuild, latestBuild, defineOverlay })
 | 
			
		||||
        ),
 | 
			
		||||
  isWDS,
 | 
			
		||||
}) => ({
 | 
			
		||||
  /**
 | 
			
		||||
   * @type { import("rollup").InputOptions }
 | 
			
		||||
   */
 | 
			
		||||
  inputOptions: {
 | 
			
		||||
    input: entry,
 | 
			
		||||
    // Some entry points contain no JavaScript. This setting silences a warning about that.
 | 
			
		||||
    // https://rollupjs.org/guide/en/#preserveentrysignatures
 | 
			
		||||
    preserveEntrySignatures: false,
 | 
			
		||||
    plugins: [
 | 
			
		||||
      ignore({
 | 
			
		||||
        files: bundle.emptyPackages({ latestBuild }),
 | 
			
		||||
      }),
 | 
			
		||||
      resolve({
 | 
			
		||||
        extensions,
 | 
			
		||||
        preferBuiltins: false,
 | 
			
		||||
        browser: true,
 | 
			
		||||
        rootDir: paths.polymer_dir,
 | 
			
		||||
      }),
 | 
			
		||||
      commonjs(),
 | 
			
		||||
      json(),
 | 
			
		||||
      babel({
 | 
			
		||||
        ...bundle.babelOptions({ latestBuild }),
 | 
			
		||||
        extensions,
 | 
			
		||||
        babelHelpers: isWDS ? "inline" : "bundled",
 | 
			
		||||
      }),
 | 
			
		||||
      string({
 | 
			
		||||
        // Import certain extensions as strings
 | 
			
		||||
        include: [path.join(paths.polymer_dir, "node_modules/**/*.css")],
 | 
			
		||||
      }),
 | 
			
		||||
      replace(bundle.definedVars({ isProdBuild, latestBuild, defineOverlay })),
 | 
			
		||||
      !isWDS &&
 | 
			
		||||
        manifest({
 | 
			
		||||
          publicPath,
 | 
			
		||||
        }),
 | 
			
		||||
        worker(),
 | 
			
		||||
        dontHashPlugin({ dontHash }),
 | 
			
		||||
        isProdBuild && terser(bundle.terserOptions(latestBuild)),
 | 
			
		||||
      !isWDS && worker(),
 | 
			
		||||
      !isWDS && dontHashPlugin({ dontHash }),
 | 
			
		||||
      !isWDS && isProdBuild && terser(bundle.terserOptions(latestBuild)),
 | 
			
		||||
      !isWDS &&
 | 
			
		||||
        isStatsBuild &&
 | 
			
		||||
          visualizer({
 | 
			
		||||
            // https://github.com/btd/rollup-plugin-visualizer#options
 | 
			
		||||
            open: true,
 | 
			
		||||
            sourcemap: true,
 | 
			
		||||
          }),
 | 
			
		||||
      ],
 | 
			
		||||
    },
 | 
			
		||||
    /**
 | 
			
		||||
     * @type { import("rollup").OutputOptions }
 | 
			
		||||
     */
 | 
			
		||||
    outputOptions: {
 | 
			
		||||
      // https://rollupjs.org/guide/en/#outputdir
 | 
			
		||||
      dir: outputPath,
 | 
			
		||||
      // https://rollupjs.org/guide/en/#outputformat
 | 
			
		||||
      format: latestBuild ? "es" : "systemjs",
 | 
			
		||||
      // https://rollupjs.org/guide/en/#outputexternallivebindings
 | 
			
		||||
      externalLiveBindings: false,
 | 
			
		||||
      // https://rollupjs.org/guide/en/#outputentryfilenames
 | 
			
		||||
      // https://rollupjs.org/guide/en/#outputchunkfilenames
 | 
			
		||||
      // https://rollupjs.org/guide/en/#outputassetfilenames
 | 
			
		||||
      entryFileNames:
 | 
			
		||||
        isProdBuild && !isStatsBuild ? "[name]-[hash].js" : "[name].js",
 | 
			
		||||
      chunkFileNames:
 | 
			
		||||
        isProdBuild && !isStatsBuild ? "c.[hash].js" : "[name].js",
 | 
			
		||||
      assetFileNames:
 | 
			
		||||
        isProdBuild && !isStatsBuild ? "a.[hash].js" : "[name].js",
 | 
			
		||||
      // https://rollupjs.org/guide/en/#outputsourcemap
 | 
			
		||||
      sourcemap: isProdBuild ? true : "inline",
 | 
			
		||||
    },
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
        visualizer({
 | 
			
		||||
          // https://github.com/btd/rollup-plugin-visualizer#options
 | 
			
		||||
          open: true,
 | 
			
		||||
          sourcemap: true,
 | 
			
		||||
        }),
 | 
			
		||||
    ].filter(Boolean),
 | 
			
		||||
  },
 | 
			
		||||
  /**
 | 
			
		||||
   * @type { import("rollup").OutputOptions }
 | 
			
		||||
   */
 | 
			
		||||
  outputOptions: {
 | 
			
		||||
    // https://rollupjs.org/guide/en/#outputdir
 | 
			
		||||
    dir: outputPath,
 | 
			
		||||
    // https://rollupjs.org/guide/en/#outputformat
 | 
			
		||||
    format: latestBuild ? "es" : "systemjs",
 | 
			
		||||
    // https://rollupjs.org/guide/en/#outputexternallivebindings
 | 
			
		||||
    externalLiveBindings: false,
 | 
			
		||||
    // https://rollupjs.org/guide/en/#outputentryfilenames
 | 
			
		||||
    // https://rollupjs.org/guide/en/#outputchunkfilenames
 | 
			
		||||
    // https://rollupjs.org/guide/en/#outputassetfilenames
 | 
			
		||||
    entryFileNames:
 | 
			
		||||
      isProdBuild && !isStatsBuild ? "[name]-[hash].js" : "[name].js",
 | 
			
		||||
    chunkFileNames: isProdBuild && !isStatsBuild ? "c.[hash].js" : "[name].js",
 | 
			
		||||
    assetFileNames: isProdBuild && !isStatsBuild ? "a.[hash].js" : "[name].js",
 | 
			
		||||
    // https://rollupjs.org/guide/en/#outputsourcemap
 | 
			
		||||
    sourcemap: isProdBuild ? true : "inline",
 | 
			
		||||
  },
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => {
 | 
			
		||||
  return createRollupConfig(
 | 
			
		||||
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild, isWDS }) =>
 | 
			
		||||
  createRollupConfig(
 | 
			
		||||
    bundle.config.app({
 | 
			
		||||
      isProdBuild,
 | 
			
		||||
      latestBuild,
 | 
			
		||||
      isStatsBuild,
 | 
			
		||||
      isWDS,
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => {
 | 
			
		||||
  return createRollupConfig(
 | 
			
		||||
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
 | 
			
		||||
  createRollupConfig(
 | 
			
		||||
    bundle.config.demo({
 | 
			
		||||
      isProdBuild,
 | 
			
		||||
      latestBuild,
 | 
			
		||||
      isStatsBuild,
 | 
			
		||||
    })
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const createCastConfig = ({ isProdBuild, latestBuild }) => {
 | 
			
		||||
  return createRollupConfig(bundle.config.cast({ isProdBuild, latestBuild }));
 | 
			
		||||
};
 | 
			
		||||
const createCastConfig = ({ isProdBuild, latestBuild }) =>
 | 
			
		||||
  createRollupConfig(bundle.config.cast({ isProdBuild, latestBuild }));
 | 
			
		||||
 | 
			
		||||
const createHassioConfig = ({ isProdBuild, latestBuild }) => {
 | 
			
		||||
  return createRollupConfig(bundle.config.hassio({ isProdBuild, latestBuild }));
 | 
			
		||||
};
 | 
			
		||||
const createHassioConfig = ({ isProdBuild, latestBuild }) =>
 | 
			
		||||
  createRollupConfig(bundle.config.hassio({ isProdBuild, latestBuild }));
 | 
			
		||||
 | 
			
		||||
const createGalleryConfig = ({ isProdBuild, latestBuild }) => {
 | 
			
		||||
  return createRollupConfig(
 | 
			
		||||
    bundle.config.gallery({ isProdBuild, latestBuild })
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
const createGalleryConfig = ({ isProdBuild, latestBuild }) =>
 | 
			
		||||
  createRollupConfig(bundle.config.gallery({ isProdBuild, latestBuild }));
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
  createAppConfig,
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,4 @@
 | 
			
		||||
/* eslint-disable @typescript-eslint/no-var-requires */
 | 
			
		||||
const path = require("path");
 | 
			
		||||
const fs = require("fs");
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,10 @@
 | 
			
		||||
/* eslint-disable @typescript-eslint/no-var-requires */
 | 
			
		||||
const webpack = require("webpack");
 | 
			
		||||
const path = require("path");
 | 
			
		||||
const TerserPlugin = require("terser-webpack-plugin");
 | 
			
		||||
const ManifestPlugin = require("webpack-manifest-plugin");
 | 
			
		||||
const { WebpackManifestPlugin } = require("webpack-manifest-plugin");
 | 
			
		||||
const paths = require("./paths.js");
 | 
			
		||||
const bundle = require("./bundle");
 | 
			
		||||
const bundle = require("./bundle.js");
 | 
			
		||||
const log = require("fancy-log");
 | 
			
		||||
 | 
			
		||||
class LogStartCompilePlugin {
 | 
			
		||||
@@ -36,6 +37,7 @@ const createWebpackConfig = ({
 | 
			
		||||
  const ignorePackages = bundle.ignorePackages({ latestBuild });
 | 
			
		||||
  return {
 | 
			
		||||
    mode: isProdBuild ? "production" : "development",
 | 
			
		||||
    target: ["web", latestBuild ? "es2017" : "es5"],
 | 
			
		||||
    devtool: isProdBuild
 | 
			
		||||
      ? "cheap-module-source-map"
 | 
			
		||||
      : "eval-cheap-module-source-map",
 | 
			
		||||
@@ -45,7 +47,6 @@ const createWebpackConfig = ({
 | 
			
		||||
      rules: [
 | 
			
		||||
        {
 | 
			
		||||
          test: /\.m?js$|\.ts$/,
 | 
			
		||||
          exclude: bundle.babelExclude(),
 | 
			
		||||
          use: {
 | 
			
		||||
            loader: "babel-loader",
 | 
			
		||||
            options: bundle.babelOptions({ latestBuild }),
 | 
			
		||||
@@ -67,7 +68,7 @@ const createWebpackConfig = ({
 | 
			
		||||
      ],
 | 
			
		||||
    },
 | 
			
		||||
    plugins: [
 | 
			
		||||
      new ManifestPlugin({
 | 
			
		||||
      new WebpackManifestPlugin({
 | 
			
		||||
        // Only include the JS of entrypoints
 | 
			
		||||
        filter: (file) => file.isInitial && !file.name.endsWith(".map"),
 | 
			
		||||
      }),
 | 
			
		||||
@@ -93,6 +94,7 @@ const createWebpackConfig = ({
 | 
			
		||||
              ? path.resolve(context, resource)
 | 
			
		||||
              : require.resolve(resource);
 | 
			
		||||
          } catch (err) {
 | 
			
		||||
            // eslint-disable-next-line no-console
 | 
			
		||||
            console.error(
 | 
			
		||||
              "Error in Home Assistant ignore plugin",
 | 
			
		||||
              resource,
 | 
			
		||||
@@ -113,8 +115,9 @@ const createWebpackConfig = ({
 | 
			
		||||
      // We need to change the import of the polyfill for EventTarget, so we replace the polyfill file with our customized one
 | 
			
		||||
      new webpack.NormalModuleReplacementPlugin(
 | 
			
		||||
        new RegExp(
 | 
			
		||||
          require.resolve(
 | 
			
		||||
            "lit-virtualizer/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js"
 | 
			
		||||
          path.resolve(
 | 
			
		||||
            paths.polymer_dir,
 | 
			
		||||
            "src/resources/lit-virtualizer/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js"
 | 
			
		||||
          )
 | 
			
		||||
        ),
 | 
			
		||||
        path.resolve(paths.polymer_dir, "src/resources/EventTarget-ponyfill.js")
 | 
			
		||||
@@ -123,6 +126,11 @@ const createWebpackConfig = ({
 | 
			
		||||
    ].filter(Boolean),
 | 
			
		||||
    resolve: {
 | 
			
		||||
      extensions: [".ts", ".js", ".json"],
 | 
			
		||||
      alias: {
 | 
			
		||||
        "lit/decorators$": "lit/decorators.js",
 | 
			
		||||
        "lit/directive$": "lit/directive.js",
 | 
			
		||||
        "lit/polyfill-support$": "lit/polyfill-support.js",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    output: {
 | 
			
		||||
      filename: ({ chunk }) => {
 | 
			
		||||
@@ -131,22 +139,6 @@ const createWebpackConfig = ({
 | 
			
		||||
        }
 | 
			
		||||
        return `${chunk.name}.${chunk.hash.substr(0, 8)}.js`;
 | 
			
		||||
      },
 | 
			
		||||
      environment: {
 | 
			
		||||
        // The environment supports arrow functions ('() => { ... }').
 | 
			
		||||
        arrowFunction: latestBuild,
 | 
			
		||||
        // The environment supports BigInt as literal (123n).
 | 
			
		||||
        bigIntLiteral: false,
 | 
			
		||||
        // The environment supports const and let for variable declarations.
 | 
			
		||||
        const: latestBuild,
 | 
			
		||||
        // The environment supports destructuring ('{ a, b } = obj').
 | 
			
		||||
        destructuring: latestBuild,
 | 
			
		||||
        // The environment supports an async import() function to import EcmaScript modules.
 | 
			
		||||
        dynamicImport: latestBuild,
 | 
			
		||||
        // The environment supports 'for of' iteration ('for (const x of array) { ... }').
 | 
			
		||||
        forOf: latestBuild,
 | 
			
		||||
        // The environment supports ECMAScript Module syntax to import ECMAScript modules (import ... from '...').
 | 
			
		||||
        module: latestBuild,
 | 
			
		||||
      },
 | 
			
		||||
      chunkFilename:
 | 
			
		||||
        isProdBuild && !isStatsBuild
 | 
			
		||||
          ? "chunk.[chunkhash].js"
 | 
			
		||||
@@ -159,33 +151,24 @@ const createWebpackConfig = ({
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => {
 | 
			
		||||
  return createWebpackConfig(
 | 
			
		||||
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
 | 
			
		||||
  createWebpackConfig(
 | 
			
		||||
    bundle.config.app({ isProdBuild, latestBuild, isStatsBuild })
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => {
 | 
			
		||||
  return createWebpackConfig(
 | 
			
		||||
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
 | 
			
		||||
  createWebpackConfig(
 | 
			
		||||
    bundle.config.demo({ isProdBuild, latestBuild, isStatsBuild })
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const createCastConfig = ({ isProdBuild, latestBuild }) => {
 | 
			
		||||
  return createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild }));
 | 
			
		||||
};
 | 
			
		||||
const createCastConfig = ({ isProdBuild, latestBuild }) =>
 | 
			
		||||
  createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild }));
 | 
			
		||||
 | 
			
		||||
const createHassioConfig = ({ isProdBuild, latestBuild }) => {
 | 
			
		||||
  return createWebpackConfig(
 | 
			
		||||
    bundle.config.hassio({ isProdBuild, latestBuild })
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
const createHassioConfig = ({ isProdBuild, latestBuild }) =>
 | 
			
		||||
  createWebpackConfig(bundle.config.hassio({ isProdBuild, latestBuild }));
 | 
			
		||||
 | 
			
		||||
const createGalleryConfig = ({ isProdBuild, latestBuild }) => {
 | 
			
		||||
  return createWebpackConfig(
 | 
			
		||||
    bundle.config.gallery({ isProdBuild, latestBuild })
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
const createGalleryConfig = ({ isProdBuild, latestBuild }) =>
 | 
			
		||||
  createWebpackConfig(bundle.config.gallery({ isProdBuild, latestBuild }));
 | 
			
		||||
 | 
			
		||||
module.exports = {
 | 
			
		||||
  createAppConfig,
 | 
			
		||||
 
 | 
			
		||||
@@ -1,16 +1,9 @@
 | 
			
		||||
import "@material/mwc-button/mwc-button";
 | 
			
		||||
import "@polymer/paper-item/paper-icon-item";
 | 
			
		||||
import "@polymer/paper-listbox/paper-listbox";
 | 
			
		||||
import { Auth, Connection } from "home-assistant-js-websocket";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, property, state } from "lit/decorators";
 | 
			
		||||
import { CastManager } from "../../../../src/cast/cast_manager";
 | 
			
		||||
import {
 | 
			
		||||
  castSendShowLovelaceView,
 | 
			
		||||
@@ -32,7 +25,6 @@ import {
 | 
			
		||||
import "../../../../src/layouts/hass-loading-screen";
 | 
			
		||||
import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config";
 | 
			
		||||
import "./hc-layout";
 | 
			
		||||
import "@material/mwc-button/mwc-button";
 | 
			
		||||
 | 
			
		||||
@customElement("hc-cast")
 | 
			
		||||
class HcCast extends LitElement {
 | 
			
		||||
@@ -42,21 +34,19 @@ class HcCast extends LitElement {
 | 
			
		||||
 | 
			
		||||
  @property() public castManager!: CastManager;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private askWrite = false;
 | 
			
		||||
  @state() private askWrite = false;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private lovelaceConfig?: LovelaceConfig | null;
 | 
			
		||||
  @state() private lovelaceConfig?: LovelaceConfig | null;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    if (this.lovelaceConfig === undefined) {
 | 
			
		||||
      return html` <hass-loading-screen no-toolbar></hass-loading-screen>> `;
 | 
			
		||||
      return html`<hass-loading-screen no-toolbar></hass-loading-screen>`;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const error =
 | 
			
		||||
      this.castManager.castState === "NO_DEVICES_AVAILABLE"
 | 
			
		||||
        ? html`
 | 
			
		||||
            <p>
 | 
			
		||||
              There were no suitable Chromecast devices to cast to found.
 | 
			
		||||
            </p>
 | 
			
		||||
            <p>There were no suitable Chromecast devices to cast to found.</p>
 | 
			
		||||
          `
 | 
			
		||||
        : undefined;
 | 
			
		||||
 | 
			
		||||
@@ -206,7 +196,7 @@ class HcCast extends LitElement {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return css`
 | 
			
		||||
      .center-item {
 | 
			
		||||
        display: flex;
 | 
			
		||||
 
 | 
			
		||||
@@ -11,15 +11,8 @@ import {
 | 
			
		||||
  getAuth,
 | 
			
		||||
  getAuthOptions,
 | 
			
		||||
} from "home-assistant-js-websocket";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, state } from "lit/decorators";
 | 
			
		||||
import { CastManager, getCastManager } from "../../../../src/cast/cast_manager";
 | 
			
		||||
import { castSendShowDemo } from "../../../../src/cast/receiver_messages";
 | 
			
		||||
import {
 | 
			
		||||
@@ -60,19 +53,19 @@ const INTRO = html`
 | 
			
		||||
 | 
			
		||||
@customElement("hc-connect")
 | 
			
		||||
export class HcConnect extends LitElement {
 | 
			
		||||
  @internalProperty() private loading = false;
 | 
			
		||||
  @state() private loading = false;
 | 
			
		||||
 | 
			
		||||
  // If we had stored credentials but we cannot connect,
 | 
			
		||||
  // show a screen asking retry or logout.
 | 
			
		||||
  @internalProperty() private cannotConnect = false;
 | 
			
		||||
  @state() private cannotConnect = false;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private error?: string | TemplateResult;
 | 
			
		||||
  @state() private error?: string | TemplateResult;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private auth?: Auth;
 | 
			
		||||
  @state() private auth?: Auth;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private connection?: Connection;
 | 
			
		||||
  @state() private connection?: Connection;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private castManager?: CastManager | null;
 | 
			
		||||
  @state() private castManager?: CastManager | null;
 | 
			
		||||
 | 
			
		||||
  private openDemo = false;
 | 
			
		||||
 | 
			
		||||
@@ -86,9 +79,7 @@ export class HcConnect extends LitElement {
 | 
			
		||||
          </div>
 | 
			
		||||
          <div class="card-actions">
 | 
			
		||||
            <a href="/">
 | 
			
		||||
              <mwc-button>
 | 
			
		||||
                Retry
 | 
			
		||||
              </mwc-button>
 | 
			
		||||
              <mwc-button> Retry </mwc-button>
 | 
			
		||||
            </a>
 | 
			
		||||
            <div class="spacer"></div>
 | 
			
		||||
            <mwc-button @click=${this._handleLogout}>Log out</mwc-button>
 | 
			
		||||
@@ -299,7 +290,7 @@ export class HcConnect extends LitElement {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return css`
 | 
			
		||||
      .card-content a {
 | 
			
		||||
        color: var(--primary-color);
 | 
			
		||||
 
 | 
			
		||||
@@ -4,15 +4,8 @@ import {
 | 
			
		||||
  getUser,
 | 
			
		||||
  HassUser,
 | 
			
		||||
} from "home-assistant-js-websocket";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, property } from "lit/decorators";
 | 
			
		||||
import "../../../../src/components/ha-card";
 | 
			
		||||
 | 
			
		||||
@customElement("hc-layout")
 | 
			
		||||
@@ -69,7 +62,7 @@ class HcLayout extends LitElement {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return css`
 | 
			
		||||
      :host {
 | 
			
		||||
        display: flex;
 | 
			
		||||
@@ -98,8 +91,12 @@ class HcLayout extends LitElement {
 | 
			
		||||
        line-height: 32px;
 | 
			
		||||
        padding: 24px 16px 16px;
 | 
			
		||||
        display: block;
 | 
			
		||||
        margin: 0;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      .hero {
 | 
			
		||||
        border-radius: 4px 4px 0 0;
 | 
			
		||||
      }
 | 
			
		||||
      .subtitle {
 | 
			
		||||
        font-size: 14px;
 | 
			
		||||
        color: var(--secondary-text-color);
 | 
			
		||||
 
 | 
			
		||||
@@ -1,10 +1,5 @@
 | 
			
		||||
import {
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  property,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { html, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, property, state } from "lit/decorators";
 | 
			
		||||
import { mockHistory } from "../../../../demo/src/stubs/history";
 | 
			
		||||
import { LovelaceConfig } from "../../../../src/data/lovelace";
 | 
			
		||||
import {
 | 
			
		||||
@@ -21,7 +16,7 @@ import "./hc-lovelace";
 | 
			
		||||
class HcDemo extends HassElement {
 | 
			
		||||
  @property({ attribute: false }) public lovelacePath!: string;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _lovelaceConfig?: LovelaceConfig;
 | 
			
		||||
  @state() private _lovelaceConfig?: LovelaceConfig;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    if (!this._lovelaceConfig) {
 | 
			
		||||
@@ -38,10 +33,10 @@ class HcDemo extends HassElement {
 | 
			
		||||
 | 
			
		||||
  protected firstUpdated(changedProps) {
 | 
			
		||||
    super.firstUpdated(changedProps);
 | 
			
		||||
    this._initialize();
 | 
			
		||||
    this._initializeHass();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _initialize() {
 | 
			
		||||
  private async _initializeHass() {
 | 
			
		||||
    const initial: Partial<MockHomeAssistant> = {
 | 
			
		||||
      // Override updateHass so that the correct hass lifecycle methods are called
 | 
			
		||||
      updateHass: (hassUpdate: Partial<HomeAssistant>) =>
 | 
			
		||||
 
 | 
			
		||||
@@ -1,12 +1,5 @@
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, property } from "lit/decorators";
 | 
			
		||||
import { HomeAssistant } from "../../../../src/types";
 | 
			
		||||
 | 
			
		||||
@customElement("hc-launch-screen")
 | 
			
		||||
@@ -29,7 +22,7 @@ class HcLaunchScreen extends LitElement {
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return css`
 | 
			
		||||
      :host {
 | 
			
		||||
        display: block;
 | 
			
		||||
 
 | 
			
		||||
@@ -1,12 +1,5 @@
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, property } from "lit/decorators";
 | 
			
		||||
import { LovelaceConfig } from "../../../../src/data/lovelace";
 | 
			
		||||
import { Lovelace } from "../../../../src/panels/lovelace/types";
 | 
			
		||||
import "../../../../src/panels/lovelace/views/hui-view";
 | 
			
		||||
@@ -35,11 +28,12 @@ class HcLovelace extends LitElement {
 | 
			
		||||
    }
 | 
			
		||||
    const lovelace: Lovelace = {
 | 
			
		||||
      config: this.lovelaceConfig,
 | 
			
		||||
      rawConfig: this.lovelaceConfig,
 | 
			
		||||
      editMode: false,
 | 
			
		||||
      urlPath: this.urlPath!,
 | 
			
		||||
      enableFullEditMode: () => undefined,
 | 
			
		||||
      mode: "storage",
 | 
			
		||||
      language: "en",
 | 
			
		||||
      locale: this.hass.locale,
 | 
			
		||||
      saveConfig: async () => undefined,
 | 
			
		||||
      deleteConfig: async () => undefined,
 | 
			
		||||
      setEditMode: () => undefined,
 | 
			
		||||
@@ -90,10 +84,11 @@ class HcLovelace extends LitElement {
 | 
			
		||||
    return undefined;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return css`
 | 
			
		||||
      :host {
 | 
			
		||||
        min-height: 100vh;
 | 
			
		||||
        height: 0;
 | 
			
		||||
        display: flex;
 | 
			
		||||
        flex-direction: column;
 | 
			
		||||
        box-sizing: border-box;
 | 
			
		||||
 
 | 
			
		||||
@@ -3,12 +3,8 @@ import {
 | 
			
		||||
  getAuth,
 | 
			
		||||
  UnsubscribeFunc,
 | 
			
		||||
} from "home-assistant-js-websocket";
 | 
			
		||||
import {
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { html, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, state } from "lit/decorators";
 | 
			
		||||
import { CAST_NS } from "../../../../src/cast/const";
 | 
			
		||||
import {
 | 
			
		||||
  ConnectMessage,
 | 
			
		||||
@@ -36,13 +32,13 @@ let resourcesLoaded = false;
 | 
			
		||||
 | 
			
		||||
@customElement("hc-main")
 | 
			
		||||
export class HcMain extends HassElement {
 | 
			
		||||
  @internalProperty() private _showDemo = false;
 | 
			
		||||
  @state() private _showDemo = false;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _lovelaceConfig?: LovelaceConfig;
 | 
			
		||||
  @state() private _lovelaceConfig?: LovelaceConfig;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _lovelacePath: string | number | null = null;
 | 
			
		||||
  @state() private _lovelacePath: string | number | null = null;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _error?: string;
 | 
			
		||||
  @state() private _error?: string;
 | 
			
		||||
 | 
			
		||||
  private _unsubLovelace?: UnsubscribeFunc;
 | 
			
		||||
 | 
			
		||||
@@ -221,11 +217,17 @@ export class HcMain extends HassElement {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _generateLovelaceConfig() {
 | 
			
		||||
    const { generateLovelaceConfigFromHass } = await import(
 | 
			
		||||
      "../../../../src/panels/lovelace/common/generate-lovelace-config"
 | 
			
		||||
    const { generateLovelaceDashboardStrategy } = await import(
 | 
			
		||||
      "../../../../src/panels/lovelace/strategies/get-strategy"
 | 
			
		||||
    );
 | 
			
		||||
    this._handleNewLovelaceConfig(
 | 
			
		||||
      await generateLovelaceConfigFromHass(this.hass!)
 | 
			
		||||
      await generateLovelaceDashboardStrategy(
 | 
			
		||||
        {
 | 
			
		||||
          hass: this.hass!,
 | 
			
		||||
          narrow: false,
 | 
			
		||||
        },
 | 
			
		||||
        "original-states"
 | 
			
		||||
      )
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
import "web-animations-js/web-animations-next-lite.min";
 | 
			
		||||
import "../../../src/resources/roboto";
 | 
			
		||||
import "../../../src/resources/ha-style";
 | 
			
		||||
import "../../../src/resources/roboto";
 | 
			
		||||
import "./layout/hc-lovelace";
 | 
			
		||||
 
 | 
			
		||||
@@ -54,6 +54,8 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
 | 
			
		||||
      state: "21",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        friendly_name: "Living room temperature",
 | 
			
		||||
        device_class: "temperature",
 | 
			
		||||
        unit_of_measurement: "°C",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    "sensor.study_temp_rounded": {
 | 
			
		||||
@@ -61,6 +63,8 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
 | 
			
		||||
      state: "23",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        friendly_name: "Study temperature",
 | 
			
		||||
        device_class: "temperature",
 | 
			
		||||
        unit_of_measurement: "°C",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    "sensor.living_room": {
 | 
			
		||||
@@ -242,11 +246,15 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
 | 
			
		||||
 | 
			
		||||
    "light.living_room_lights": {
 | 
			
		||||
      entity_id: "light.living_room_lights",
 | 
			
		||||
      state: "off",
 | 
			
		||||
      state: "on",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        min_mireds: 111,
 | 
			
		||||
        max_mireds: 400,
 | 
			
		||||
        brightness: 175,
 | 
			
		||||
        color_temp: 300,
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp"],
 | 
			
		||||
        friendly_name: "Living Room Lights",
 | 
			
		||||
        color_mode: "color_temp",
 | 
			
		||||
        supported_features: 55,
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
@@ -259,13 +267,27 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
 | 
			
		||||
    },
 | 
			
		||||
    "light.kitchen_lights": {
 | 
			
		||||
      entity_id: "light.kitchen_lights",
 | 
			
		||||
      state: "on",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        min_mireds: 111,
 | 
			
		||||
        max_mireds: 400,
 | 
			
		||||
        brightness: 200,
 | 
			
		||||
        rgb_color: [255, 175, 96],
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp", "rgb"],
 | 
			
		||||
        color_mode: "rgb",
 | 
			
		||||
        friendly_name: "Kitchen Lights",
 | 
			
		||||
        supported_features: 55,
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    "light.lifx5": {
 | 
			
		||||
      entity_id: "light.lifx5",
 | 
			
		||||
      state: "off",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        friendly_name: "Kitchen lights",
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        friendly_name: "Garage Lights",
 | 
			
		||||
        supported_features: 1,
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    "sensor.plexspy": {
 | 
			
		||||
      entity_id: "sensor.plexspy",
 | 
			
		||||
      state: "0",
 | 
			
		||||
@@ -478,16 +500,6 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
 | 
			
		||||
        icon: "hademo:history",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    "light.lifx5": {
 | 
			
		||||
      entity_id: "light.lifx5",
 | 
			
		||||
      state: "on",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        min_mireds: 111,
 | 
			
		||||
        max_mireds: 400,
 | 
			
		||||
        friendly_name: "Garage lights",
 | 
			
		||||
        supported_features: 55,
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    "sensor.alok_to_home": {
 | 
			
		||||
      entity_id: "sensor.alok_to_home",
 | 
			
		||||
      state: "41",
 | 
			
		||||
 
 | 
			
		||||
@@ -12,6 +12,7 @@ export const demoLovelaceArsaboo: DemoConfig["lovelace"] = (localize) => ({
 | 
			
		||||
        {
 | 
			
		||||
          type: "entities",
 | 
			
		||||
          title: localize("ui.panel.page-demo.config.arsaboo.labels.lights"),
 | 
			
		||||
          state_color: true,
 | 
			
		||||
          entities: [
 | 
			
		||||
            {
 | 
			
		||||
              entity: "light.kitchen_lights",
 | 
			
		||||
 
 | 
			
		||||
@@ -3,22 +3,10 @@ import { Lovelace } from "../../../src/panels/lovelace/types";
 | 
			
		||||
import { DemoConfig } from "./types";
 | 
			
		||||
 | 
			
		||||
export const demoConfigs: Array<() => Promise<DemoConfig>> = [
 | 
			
		||||
  () =>
 | 
			
		||||
    import(/* webpackChunkName: "arsaboo" */ "./arsaboo").then(
 | 
			
		||||
      (mod) => mod.demoArsaboo
 | 
			
		||||
    ),
 | 
			
		||||
  () =>
 | 
			
		||||
    import(/* webpackChunkName: "teachingbirds" */ "./teachingbirds").then(
 | 
			
		||||
      (mod) => mod.demoTeachingbirds
 | 
			
		||||
    ),
 | 
			
		||||
  () =>
 | 
			
		||||
    import(/* webpackChunkName: "kernehed" */ "./kernehed").then(
 | 
			
		||||
      (mod) => mod.demoKernehed
 | 
			
		||||
    ),
 | 
			
		||||
  () =>
 | 
			
		||||
    import(/* webpackChunkName: "jimpower" */ "./jimpower").then(
 | 
			
		||||
      (mod) => mod.demoJimpower
 | 
			
		||||
    ),
 | 
			
		||||
  () => import("./arsaboo").then((mod) => mod.demoArsaboo),
 | 
			
		||||
  () => import("./teachingbirds").then((mod) => mod.demoTeachingbirds),
 | 
			
		||||
  () => import("./kernehed").then((mod) => mod.demoKernehed),
 | 
			
		||||
  () => import("./jimpower").then((mod) => mod.demoJimpower),
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
// eslint-disable-next-line import/no-mutable-exports
 | 
			
		||||
 
 | 
			
		||||
@@ -653,7 +653,7 @@ export const demoEntitiesJimpower: DemoConfig["entities"] = () =>
 | 
			
		||||
      entity_id: "binary_sensor.smoke_sensor_158d0001b8ddc7",
 | 
			
		||||
      state: "off",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        Density: 0,
 | 
			
		||||
        density: 0,
 | 
			
		||||
        battery_level: 59,
 | 
			
		||||
        friendly_name: "Downstairs Smoke Detector",
 | 
			
		||||
        device_class: "smoke",
 | 
			
		||||
@@ -663,7 +663,7 @@ export const demoEntitiesJimpower: DemoConfig["entities"] = () =>
 | 
			
		||||
      entity_id: "binary_sensor.smoke_sensor_158d0001b8deba",
 | 
			
		||||
      state: "off",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        Density: 0,
 | 
			
		||||
        density: 0,
 | 
			
		||||
        battery_level: 65,
 | 
			
		||||
        friendly_name: "Upstairs Smoke Detector",
 | 
			
		||||
        device_class: "smoke",
 | 
			
		||||
 
 | 
			
		||||
@@ -3,49 +3,7 @@ import { DemoConfig } from "../types";
 | 
			
		||||
 | 
			
		||||
export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
  name: "Kingia Castle",
 | 
			
		||||
  resources: [
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/dark-sky-weather-card.js?v=4",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/mini-media-player-bundle.js?v=0.9.8",
 | 
			
		||||
    //   type: "module",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/tracker-card.js?v=0.1.5",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/surveillance-card.js?v=0.0.1",
 | 
			
		||||
    //   type: "module",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/mini-graph-card-bundle.js?v=0.1.0",
 | 
			
		||||
    //   type: "module",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/slider-entity-row.js?v=d6da75",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url:
 | 
			
		||||
    //     "/local/custom_ui/compact-custom-header/compact-custom-header.js?v=0.2.7",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/waze-card.js?v=1.1.1",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/circle-sensor-card.js?v=1.2.0",
 | 
			
		||||
    //   type: "module",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom_ui/monster-card.js?v=0.2.3",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
  ],
 | 
			
		||||
  resources: [],
 | 
			
		||||
  views: [
 | 
			
		||||
    {
 | 
			
		||||
      cards: [
 | 
			
		||||
@@ -603,89 +561,6 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          cards: [
 | 
			
		||||
            // {
 | 
			
		||||
            //   style: {
 | 
			
		||||
            //     "background-image": 'url("/assets/jimpower/cardbackK.png")',
 | 
			
		||||
            //     "background-size": "100% 400px",
 | 
			
		||||
            //     "box-shadow": "3px 3px rgba(0,0,0,0.4)",
 | 
			
		||||
            //     "background-repeat": "no-repeat",
 | 
			
		||||
            //     color: "#999999",
 | 
			
		||||
            //     "border-radius": "20px",
 | 
			
		||||
            //     border: "solid 1px rgba(100,100,100,0.3)",
 | 
			
		||||
            //     "background-color": "rgba(50,50,50,0.3)",
 | 
			
		||||
            //   },
 | 
			
		||||
            //   type: "custom:card-modder",
 | 
			
		||||
            //   card: {
 | 
			
		||||
            //     entity_visibility: "sensor.dark_sky_visibility",
 | 
			
		||||
            //     entity_sun: "sun.sun",
 | 
			
		||||
            //     entity_daily_summary:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_detailed_summary_0",
 | 
			
		||||
            //     entity_temperature: "sensor.bom_temp",
 | 
			
		||||
            //     entity_forecast_high_temp_3:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_max_temp_c_3",
 | 
			
		||||
            //     entity_forecast_high_temp_2:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_max_temp_c_2",
 | 
			
		||||
            //     entity_forecast_high_temp_5:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_max_temp_c_5",
 | 
			
		||||
            //     entity_forecast_high_temp_4:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_max_temp_c_4",
 | 
			
		||||
            //     entity_wind_speed: "sensor.bom_wind_sp",
 | 
			
		||||
            //     entity_forecast_icon_4: "sensor.dark_sky_icon_4",
 | 
			
		||||
            //     entity_forecast_icon_5: "sensor.dark_sky_icon_5",
 | 
			
		||||
            //     entity_forecast_icon_2: "sensor.dark_sky_icon_2",
 | 
			
		||||
            //     entity_forecast_icon_3: "sensor.dark_sky_icon_3",
 | 
			
		||||
            //     entity_forecast_icon_1: "sensor.dark_sky_icon_1",
 | 
			
		||||
            //     entity_forecast_high_temp_1:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_max_temp_c_1",
 | 
			
		||||
            //     entity_wind_bearing: "sensor.bom_wind_bear",
 | 
			
		||||
            //     entity_forecast_low_temp_2:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_min_temp_c_2",
 | 
			
		||||
            //     entity_forecast_low_temp_3:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_min_temp_c_3",
 | 
			
		||||
            //     entity_pressure: "sensor.bom_pres",
 | 
			
		||||
            //     entity_forecast_low_temp_1:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_min_temp_c_1",
 | 
			
		||||
            //     entity_forecast_low_temp_4:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_min_temp_c_4",
 | 
			
		||||
            //     entity_forecast_low_temp_5:
 | 
			
		||||
            //       "sensor.bom_gc_forecast_min_temp_c_5",
 | 
			
		||||
            //     entity_humidity: "sensor.bom_humd",
 | 
			
		||||
            //     type: "custom:dark-sky-weather-card",
 | 
			
		||||
            //     entity_current_conditions: "sensor.dark_sky_icon",
 | 
			
		||||
            //   },
 | 
			
		||||
            // },
 | 
			
		||||
            // {
 | 
			
		||||
            //   style: {
 | 
			
		||||
            //     "background-image": 'url("/assets/jimpower/home/waze_5.png")',
 | 
			
		||||
            //     "background-size": "100% 400px",
 | 
			
		||||
            //     "box-shadow": "3px 3px rgba(0,0,0,0.4)",
 | 
			
		||||
            //     "background-repeat": "no-repeat",
 | 
			
		||||
            //     "border-radius": "20px",
 | 
			
		||||
            //     border: "solid 1px rgba(100,100,100,0.3)",
 | 
			
		||||
            //     "background-color": "rgba(50,50,50,0.3)",
 | 
			
		||||
            //   },
 | 
			
		||||
            //   type: "custom:card-modder",
 | 
			
		||||
            //   card: {
 | 
			
		||||
            //     entities: [
 | 
			
		||||
            //       {
 | 
			
		||||
            //         name: "James",
 | 
			
		||||
            //         zone: "zone.home",
 | 
			
		||||
            //         entity: "sensor.james_to_home",
 | 
			
		||||
            //       },
 | 
			
		||||
            //       {
 | 
			
		||||
            //         name: "Tina",
 | 
			
		||||
            //         zone: "zone.home",
 | 
			
		||||
            //         entity: "sensor.tina_to_home",
 | 
			
		||||
            //       },
 | 
			
		||||
            //       {
 | 
			
		||||
            //         name: "Work",
 | 
			
		||||
            //         zone: "zone.powertec",
 | 
			
		||||
            //         entity: "sensor.commute_to_work",
 | 
			
		||||
            //       },
 | 
			
		||||
            //     ],
 | 
			
		||||
            //     type: "custom:waze-card",
 | 
			
		||||
            //   },
 | 
			
		||||
            // },
 | 
			
		||||
            {
 | 
			
		||||
              style: {
 | 
			
		||||
                "border-radius": "20px",
 | 
			
		||||
@@ -722,46 +597,8 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          ],
 | 
			
		||||
          type: "vertical-stack",
 | 
			
		||||
        },
 | 
			
		||||
        // {
 | 
			
		||||
        //   cards: [
 | 
			
		||||
        //     {
 | 
			
		||||
        //       style: {
 | 
			
		||||
        //         "border-radius": "20px",
 | 
			
		||||
        //         color: "#999999",
 | 
			
		||||
        //         "box-shadow": "3px 3px rgba(0,0,0,0.4)",
 | 
			
		||||
        //         border: "solid 1px rgba(100,100,100,0.3)",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       type: "custom:card-modder",
 | 
			
		||||
        //       card: {
 | 
			
		||||
        //         type: "picture-entity",
 | 
			
		||||
        //         entity: "camera.bom_radar",
 | 
			
		||||
        //       },
 | 
			
		||||
        //     },
 | 
			
		||||
        //     // {
 | 
			
		||||
        //     //   style: {
 | 
			
		||||
        //     //     "background-image": 'url("/assets/jimpower/cardbackK.png")',
 | 
			
		||||
        //     //     "background-size": "100% 525px",
 | 
			
		||||
        //     //     "box-shadow": "3px 3px rgba(0,0,0,0.4)",
 | 
			
		||||
        //     //     "background-repeat": "no-repeat",
 | 
			
		||||
        //     //     color: "#999999",
 | 
			
		||||
        //     //     "border-radius": "20px",
 | 
			
		||||
        //     //     border: "solid 1px rgba(100,100,100,0.3)",
 | 
			
		||||
        //     //     "background-color": "rgba(50,50,50,0.3)",
 | 
			
		||||
        //     //   },
 | 
			
		||||
        //     //   type: "custom:card-modder",
 | 
			
		||||
        //     //   card: {
 | 
			
		||||
        //     //     title: null,
 | 
			
		||||
        //     //     type: "custom:tracker-card",
 | 
			
		||||
        //     //     trackers: [
 | 
			
		||||
        //     //       "sensor.custom_card_tracker",
 | 
			
		||||
        //     //       "sensor.custom_component_tracker",
 | 
			
		||||
        //     //     ],
 | 
			
		||||
        //     //   },
 | 
			
		||||
        //     // },
 | 
			
		||||
        //   ],
 | 
			
		||||
        //   type: "vertical-stack",
 | 
			
		||||
        // },
 | 
			
		||||
      ],
 | 
			
		||||
      path: "home",
 | 
			
		||||
      icon: "mdi:castle",
 | 
			
		||||
      name: "Home",
 | 
			
		||||
      background:
 | 
			
		||||
@@ -881,26 +718,13 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
              card: {
 | 
			
		||||
                image: "/assets/jimpower/security/air_8.jpg",
 | 
			
		||||
                elements: [
 | 
			
		||||
                  {
 | 
			
		||||
                    image:
 | 
			
		||||
                      "https://www.airvisual.com/assets/aqi/ic-face-1-green.svg",
 | 
			
		||||
                    type: "image",
 | 
			
		||||
                    style: {
 | 
			
		||||
                      width: "80px",
 | 
			
		||||
                      top: "30%",
 | 
			
		||||
                      left: "12%",
 | 
			
		||||
                      transform: "none",
 | 
			
		||||
                      height: "80px",
 | 
			
		||||
                    },
 | 
			
		||||
                    entity: "sensor.us_air_pollution_level_2",
 | 
			
		||||
                  },
 | 
			
		||||
                  {
 | 
			
		||||
                    style: {
 | 
			
		||||
                      color: "hsl(120, 41%, 39%)",
 | 
			
		||||
                      top: "50%",
 | 
			
		||||
                      "font-weight": 600,
 | 
			
		||||
                      "font-size": "20px",
 | 
			
		||||
                      left: "44%",
 | 
			
		||||
                      "font-size": "50px",
 | 
			
		||||
                      left: "30%",
 | 
			
		||||
                    },
 | 
			
		||||
                    type: "state-label",
 | 
			
		||||
                    entity: "sensor.us_air_pollution_level_2",
 | 
			
		||||
@@ -920,7 +744,7 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
                    style: {
 | 
			
		||||
                      color: "white",
 | 
			
		||||
                      top: "80%",
 | 
			
		||||
                      left: "52%",
 | 
			
		||||
                      left: "48%",
 | 
			
		||||
                    },
 | 
			
		||||
                    type: "state-icon",
 | 
			
		||||
                    entity: "sensor.us_main_pollutant_2",
 | 
			
		||||
@@ -1411,6 +1235,7 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          type: "vertical-stack",
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      path: "security",
 | 
			
		||||
      icon: "hass:shield-home",
 | 
			
		||||
      name: "Security",
 | 
			
		||||
      background:
 | 
			
		||||
 
 | 
			
		||||
@@ -101,7 +101,12 @@ export const demoEntitiesKernehed: DemoConfig["entities"] = () =>
 | 
			
		||||
    "sensor.zwave_battery_front_door": {
 | 
			
		||||
      entity_id: "sensor.zwave_battery_front_door",
 | 
			
		||||
      state: "63",
 | 
			
		||||
      attributes: { friendly_name: "Battery", icon: "mdi:battery-60" },
 | 
			
		||||
      attributes: {
 | 
			
		||||
        friendly_name: "Battery",
 | 
			
		||||
        icon: "mdi:battery-60",
 | 
			
		||||
        unit_of_measurement: "%",
 | 
			
		||||
        device_class: "battery",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    "sensor.oskar_devices": {
 | 
			
		||||
      entity_id: "sensor.oskar_devices",
 | 
			
		||||
@@ -164,7 +169,7 @@ export const demoEntitiesKernehed: DemoConfig["entities"] = () =>
 | 
			
		||||
    },
 | 
			
		||||
    "input_select.christmas_pattern": {
 | 
			
		||||
      entity_id: "input_select.christmas_pattern",
 | 
			
		||||
      state: "None",
 | 
			
		||||
      state: "Rainbow",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        options: [
 | 
			
		||||
          "None",
 | 
			
		||||
@@ -186,7 +191,7 @@ export const demoEntitiesKernehed: DemoConfig["entities"] = () =>
 | 
			
		||||
    },
 | 
			
		||||
    "input_select.christmas_palette": {
 | 
			
		||||
      entity_id: "input_select.christmas_palette",
 | 
			
		||||
      state: "None",
 | 
			
		||||
      state: "Party",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        options: [
 | 
			
		||||
          "None",
 | 
			
		||||
@@ -457,7 +462,7 @@ export const demoEntitiesKernehed: DemoConfig["entities"] = () =>
 | 
			
		||||
      state: "0.0",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        unit_of_measurement: "kB/s",
 | 
			
		||||
        friendly_name: "Nedladdning",
 | 
			
		||||
        friendly_name: "Downloading",
 | 
			
		||||
        icon: "mdi:file-download",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
@@ -471,7 +476,7 @@ export const demoEntitiesKernehed: DemoConfig["entities"] = () =>
 | 
			
		||||
      state: "0.0",
 | 
			
		||||
      attributes: {
 | 
			
		||||
        unit_of_measurement: "kB/s",
 | 
			
		||||
        friendly_name: "Uppladdning",
 | 
			
		||||
        friendly_name: "Uploading",
 | 
			
		||||
        icon: "mdi:file-upload",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
 
 | 
			
		||||
@@ -2,44 +2,7 @@ import { DemoConfig } from "../types";
 | 
			
		||||
 | 
			
		||||
export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
  name: "Hem",
 | 
			
		||||
  resources: [
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/monster-card.js",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/mini-media-player-bundle.js?v=0.9.8",
 | 
			
		||||
    //   type: "module",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/slideshow-card.js?=1.1.0",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/fold-entity-row.js?v=3ae2c4",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/swipe-card/swipe-card.js?v=2.0.0",
 | 
			
		||||
    //   type: "module",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/upcoming-media-card/upcoming-media-card.js",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/tracker-card.js?v=0.1.5",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/card-tools.js?v=6ce5d0",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
    // {
 | 
			
		||||
    //   url: "/local/custom-lovelace/krisinfo.js?=0.0.1",
 | 
			
		||||
    //   type: "js",
 | 
			
		||||
    // },
 | 
			
		||||
  ],
 | 
			
		||||
  resources: [],
 | 
			
		||||
  views: [
 | 
			
		||||
    {
 | 
			
		||||
      cards: [
 | 
			
		||||
@@ -64,7 +27,7 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
                      style: {
 | 
			
		||||
                        color: "white",
 | 
			
		||||
                        top: "93%",
 | 
			
		||||
                        left: "90%",
 | 
			
		||||
                        left: "85%",
 | 
			
		||||
                      },
 | 
			
		||||
                      type: "state-label",
 | 
			
		||||
                      entity: "sensor.battery_oskar",
 | 
			
		||||
@@ -87,7 +50,7 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
                    {
 | 
			
		||||
                      style: {
 | 
			
		||||
                        color: "white",
 | 
			
		||||
                        top: "92%",
 | 
			
		||||
                        top: "93%",
 | 
			
		||||
                        left: "20%",
 | 
			
		||||
                      },
 | 
			
		||||
                      type: "state-label",
 | 
			
		||||
@@ -96,8 +59,8 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
                    {
 | 
			
		||||
                      style: {
 | 
			
		||||
                        color: "white",
 | 
			
		||||
                        top: "92%",
 | 
			
		||||
                        left: "90%",
 | 
			
		||||
                        top: "93%",
 | 
			
		||||
                        left: "85%",
 | 
			
		||||
                      },
 | 
			
		||||
                      type: "state-label",
 | 
			
		||||
                      entity: "sensor.battery_bella",
 | 
			
		||||
@@ -105,7 +68,7 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
                    {
 | 
			
		||||
                      style: {
 | 
			
		||||
                        color: "white",
 | 
			
		||||
                        top: "92%",
 | 
			
		||||
                        top: "93%",
 | 
			
		||||
                        left: "55%",
 | 
			
		||||
                      },
 | 
			
		||||
                      type: "state-label",
 | 
			
		||||
@@ -131,78 +94,6 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          type: "entities",
 | 
			
		||||
          title: "Lock",
 | 
			
		||||
        },
 | 
			
		||||
        // {
 | 
			
		||||
        //   filter: {
 | 
			
		||||
        //     exclude: [
 | 
			
		||||
        //       {
 | 
			
		||||
        //         state: "not_home",
 | 
			
		||||
        //       },
 | 
			
		||||
        //     ],
 | 
			
		||||
        //     include: [
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "device_tracker.annasiphone",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "device_tracker.iphone_2",
 | 
			
		||||
        //       },
 | 
			
		||||
        //     ],
 | 
			
		||||
        //   },
 | 
			
		||||
        //   type: "custom:monster-card",
 | 
			
		||||
        //   card: {
 | 
			
		||||
        //     show_header_toggle: false,
 | 
			
		||||
        //     type: "entities",
 | 
			
		||||
        //     title: "G\u00e4ster",
 | 
			
		||||
        //   },
 | 
			
		||||
        //   show_empty: false,
 | 
			
		||||
        // },
 | 
			
		||||
        // {
 | 
			
		||||
        //   filter: {
 | 
			
		||||
        //     exclude: [
 | 
			
		||||
        //       {
 | 
			
		||||
        //         state: "Inget",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         state: "i.u.",
 | 
			
		||||
        //       },
 | 
			
		||||
        //     ],
 | 
			
		||||
        //     include: [
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_al",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_alm",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_salg_vide",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_bjork",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_bok",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_ek",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_grabo",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_gras",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id: "sensor.pollen_hassel",
 | 
			
		||||
        //       },
 | 
			
		||||
        //     ],
 | 
			
		||||
        //   },
 | 
			
		||||
        //   type: "custom:monster-card",
 | 
			
		||||
        //   card: {
 | 
			
		||||
        //     show_header_toggle: false,
 | 
			
		||||
        //     type: "entities",
 | 
			
		||||
        //     title: "Pollenniv\u00e5er",
 | 
			
		||||
        //   },
 | 
			
		||||
        //   show_empty: false,
 | 
			
		||||
        // },
 | 
			
		||||
        {
 | 
			
		||||
          cards: [
 | 
			
		||||
            {
 | 
			
		||||
@@ -226,10 +117,6 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          ],
 | 
			
		||||
          type: "vertical-stack",
 | 
			
		||||
        },
 | 
			
		||||
        // {
 | 
			
		||||
        //   url: "https://embed.windy.com/embed2.html",
 | 
			
		||||
        //   type: "iframe",
 | 
			
		||||
        // },
 | 
			
		||||
        {
 | 
			
		||||
          entities: [
 | 
			
		||||
            {
 | 
			
		||||
@@ -263,6 +150,7 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          ],
 | 
			
		||||
          type: "glance",
 | 
			
		||||
          show_state: false,
 | 
			
		||||
          columns: 4,
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          entities: ["sensor.oskar_bluetooth"],
 | 
			
		||||
@@ -270,32 +158,6 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          type: "entities",
 | 
			
		||||
          title: "Occupancy",
 | 
			
		||||
        },
 | 
			
		||||
        // {
 | 
			
		||||
        //   filter: {
 | 
			
		||||
        //     exclude: [
 | 
			
		||||
        //       {
 | 
			
		||||
        //         state: false,
 | 
			
		||||
        //       },
 | 
			
		||||
        //     ],
 | 
			
		||||
        //     include: [
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id:
 | 
			
		||||
        //           "binary_sensor.fibaro_system_unknown_type0c02_id1003_sensor_2",
 | 
			
		||||
        //       },
 | 
			
		||||
        //       {
 | 
			
		||||
        //         entity_id:
 | 
			
		||||
        //           "binary_sensor.fibaro_system_unknown_type0c02_id1003_sensor_3",
 | 
			
		||||
        //       },
 | 
			
		||||
        //     ],
 | 
			
		||||
        //   },
 | 
			
		||||
        //   type: "custom:monster-card",
 | 
			
		||||
        //   card: {
 | 
			
		||||
        //     show_header_toggle: false,
 | 
			
		||||
        //     type: "entities",
 | 
			
		||||
        //     title: "Brandvarnare",
 | 
			
		||||
        //   },
 | 
			
		||||
        //   show_empty: false,
 | 
			
		||||
        // },
 | 
			
		||||
        {
 | 
			
		||||
          type: "weather-forecast",
 | 
			
		||||
          entity: "weather.smhi_vader",
 | 
			
		||||
@@ -378,41 +240,9 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
            "binary_sensor.windows_server",
 | 
			
		||||
            "binary_sensor.teamspeak",
 | 
			
		||||
            "binary_sensor.harmony_hub",
 | 
			
		||||
            // {
 | 
			
		||||
            //   style: {
 | 
			
		||||
            //     height: "1px",
 | 
			
		||||
            //     width: "85%",
 | 
			
		||||
            //     "margin-left": "auto",
 | 
			
		||||
            //     background: "#62717b",
 | 
			
		||||
            //     "margin-right": "auto",
 | 
			
		||||
            //   },
 | 
			
		||||
            //   type: "divider",
 | 
			
		||||
            // },
 | 
			
		||||
            // {
 | 
			
		||||
            //   items: ["sensor.uptime_router", "sensor.installerad_routeros"],
 | 
			
		||||
            //   head: {
 | 
			
		||||
            //     entity: "binary_sensor.router",
 | 
			
		||||
            //   },
 | 
			
		||||
            //   type: "custom:fold-entity-row",
 | 
			
		||||
            //   group_config: {
 | 
			
		||||
            //     icon: "mdi:router",
 | 
			
		||||
            //   },
 | 
			
		||||
            // },
 | 
			
		||||
            // {
 | 
			
		||||
            //   items: [
 | 
			
		||||
            //     "sensor.uptime_router_server",
 | 
			
		||||
            //     "sensor.installerad_routeros_server",
 | 
			
		||||
            //   ],
 | 
			
		||||
            //   head: {
 | 
			
		||||
            //     entity: "binary_sensor.router_server",
 | 
			
		||||
            //   },
 | 
			
		||||
            //   type: "custom:fold-entity-row",
 | 
			
		||||
            //   group_config: {
 | 
			
		||||
            //     icon: "mdi:router",
 | 
			
		||||
            //   },
 | 
			
		||||
            // },
 | 
			
		||||
          ],
 | 
			
		||||
          show_header_toggle: false,
 | 
			
		||||
          state_color: true,
 | 
			
		||||
          type: "entities",
 | 
			
		||||
          title: "Network",
 | 
			
		||||
        },
 | 
			
		||||
@@ -422,29 +252,10 @@ export const demoLovelaceKernehed: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
            "binary_sensor.ubiquiti_switch",
 | 
			
		||||
            "binary_sensor.ubiquiti_nvr",
 | 
			
		||||
            "binary_sensor.entre_kamera",
 | 
			
		||||
            // {
 | 
			
		||||
            //   items: ["sensor.uptime_ap_1"],
 | 
			
		||||
            //   head: {
 | 
			
		||||
            //     entity: "binary_sensor.accesspunkt_1",
 | 
			
		||||
            //   },
 | 
			
		||||
            //   type: "custom:fold-entity-row",
 | 
			
		||||
            //   group_config: {
 | 
			
		||||
            //     icon: "router-wireless",
 | 
			
		||||
            //   },
 | 
			
		||||
            // },
 | 
			
		||||
            // {
 | 
			
		||||
            //   items: ["sensor.uptime_ap_2"],
 | 
			
		||||
            //   head: {
 | 
			
		||||
            //     entity: "binary_sensor.accesspunkt_2",
 | 
			
		||||
            //   },
 | 
			
		||||
            //   type: "custom:fold-entity-row",
 | 
			
		||||
            //   group_config: {
 | 
			
		||||
            //     icon: "router-wireless",
 | 
			
		||||
            //   },
 | 
			
		||||
            // },
 | 
			
		||||
            "sensor.total_clients_wireless",
 | 
			
		||||
          ],
 | 
			
		||||
          show_header_toggle: false,
 | 
			
		||||
          state_color: true,
 | 
			
		||||
          type: "entities",
 | 
			
		||||
          title: "Ubiquiti",
 | 
			
		||||
        },
 | 
			
		||||
 
 | 
			
		||||
@@ -1114,6 +1114,9 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        min_mireds: 153,
 | 
			
		||||
        max_mireds: 500,
 | 
			
		||||
        brightness: 63,
 | 
			
		||||
        color_temp: 200,
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp", "rgb"],
 | 
			
		||||
        color_mode: "color_temp",
 | 
			
		||||
        friendly_name: "Upstairs lights",
 | 
			
		||||
        supported_features: 63,
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
@@ -1125,6 +1128,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
      attributes: {
 | 
			
		||||
        friendly_name: "Walk in closet lights",
 | 
			
		||||
        supported_features: 41,
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp"],
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:wall-sconce",
 | 
			
		||||
      },
 | 
			
		||||
@@ -1136,6 +1140,8 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        brightness: 254,
 | 
			
		||||
        friendly_name: "Outdoor lights",
 | 
			
		||||
        supported_features: 41,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        color_mode: "brightness",
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:wall-sconce",
 | 
			
		||||
      },
 | 
			
		||||
@@ -1148,6 +1154,8 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        max_mireds: 500,
 | 
			
		||||
        brightness: 128,
 | 
			
		||||
        color_temp: 366,
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp", "rgb"],
 | 
			
		||||
        color_mode: "color_temp",
 | 
			
		||||
        effect_list: ["colorloop"],
 | 
			
		||||
        friendly_name: "Downstairs lights",
 | 
			
		||||
        supported_features: 63,
 | 
			
		||||
@@ -1307,6 +1315,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
      attributes: {
 | 
			
		||||
        min_mireds: 153,
 | 
			
		||||
        max_mireds: 500,
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp"],
 | 
			
		||||
        is_deconz_group: false,
 | 
			
		||||
        friendly_name: "Bedside Lamp",
 | 
			
		||||
        supported_features: 63,
 | 
			
		||||
@@ -1320,6 +1329,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
      attributes: {
 | 
			
		||||
        min_mireds: 153,
 | 
			
		||||
        max_mireds: 500,
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp"],
 | 
			
		||||
        is_deconz_group: false,
 | 
			
		||||
        friendly_name: "Floorlamp Reading Light",
 | 
			
		||||
        supported_features: 43,
 | 
			
		||||
@@ -1335,6 +1345,8 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        max_mireds: 500,
 | 
			
		||||
        brightness: 128,
 | 
			
		||||
        color_temp: 366,
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp", "rgb"],
 | 
			
		||||
        color_mode: "color_temp",
 | 
			
		||||
        effect_list: ["colorloop"],
 | 
			
		||||
        is_deconz_group: false,
 | 
			
		||||
        friendly_name: "Hallway window light",
 | 
			
		||||
@@ -1349,6 +1361,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
      attributes: {
 | 
			
		||||
        brightness: 77,
 | 
			
		||||
        is_deconz_group: false,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        friendly_name: "Isa Ceiling Light",
 | 
			
		||||
        supported_features: 41,
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
@@ -1363,6 +1376,8 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        max_mireds: 500,
 | 
			
		||||
        brightness: 150,
 | 
			
		||||
        color_temp: 366,
 | 
			
		||||
        supported_color_modes: ["brightness", "color_temp"],
 | 
			
		||||
        color_mode: "color_temp",
 | 
			
		||||
        effect_list: ["colorloop"],
 | 
			
		||||
        is_deconz_group: false,
 | 
			
		||||
        friendly_name: "Floorlamp",
 | 
			
		||||
@@ -1377,6 +1392,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
      attributes: {
 | 
			
		||||
        friendly_name: "Bedroom Ceiling Light",
 | 
			
		||||
        supported_features: 41,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:ceiling-light",
 | 
			
		||||
      },
 | 
			
		||||
@@ -1387,6 +1403,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
      attributes: {
 | 
			
		||||
        friendly_name: "Nightlight",
 | 
			
		||||
        supported_features: 17,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:lamp",
 | 
			
		||||
      },
 | 
			
		||||
@@ -1753,6 +1770,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        power_consumption: 2.2,
 | 
			
		||||
        friendly_name: "Upstairs Hallway Light",
 | 
			
		||||
        supported_features: 33,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:ceiling-light",
 | 
			
		||||
      },
 | 
			
		||||
@@ -1768,6 +1786,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        power_consumption: 0,
 | 
			
		||||
        friendly_name: "Dining Room Light",
 | 
			
		||||
        supported_features: 33,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:ceiling-light",
 | 
			
		||||
      },
 | 
			
		||||
@@ -1783,6 +1802,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        power_consumption: 0,
 | 
			
		||||
        friendly_name: "Living room Spotlights",
 | 
			
		||||
        supported_features: 33,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:track-light",
 | 
			
		||||
      },
 | 
			
		||||
@@ -1799,6 +1819,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        power_consumption: 2.5,
 | 
			
		||||
        friendly_name: "Passage Lights",
 | 
			
		||||
        supported_features: 33,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:track-light",
 | 
			
		||||
      },
 | 
			
		||||
@@ -1843,6 +1864,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
 | 
			
		||||
        power_consumption: 37.4,
 | 
			
		||||
        friendly_name: "Kitchen Lights",
 | 
			
		||||
        supported_features: 33,
 | 
			
		||||
        supported_color_modes: ["brightness"],
 | 
			
		||||
        custom_ui_state_card: "state-card-custom-ui",
 | 
			
		||||
        icon: "mdi:track-light",
 | 
			
		||||
      },
 | 
			
		||||
 
 | 
			
		||||
@@ -215,6 +215,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          card: {
 | 
			
		||||
            type: "glance",
 | 
			
		||||
            show_state: false,
 | 
			
		||||
            columns: 4,
 | 
			
		||||
          },
 | 
			
		||||
          state_filter: ["on"],
 | 
			
		||||
        },
 | 
			
		||||
@@ -439,57 +440,43 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          type: "horizontal-stack",
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          type: "grid",
 | 
			
		||||
          columns: 2,
 | 
			
		||||
          cards: [
 | 
			
		||||
            {
 | 
			
		||||
              cards: [
 | 
			
		||||
                {
 | 
			
		||||
                  graph: "line",
 | 
			
		||||
                  type: "sensor",
 | 
			
		||||
                  entity: "sensor.temperature_bedroom",
 | 
			
		||||
                },
 | 
			
		||||
                {
 | 
			
		||||
                  graph: "line",
 | 
			
		||||
                  type: "sensor",
 | 
			
		||||
                  name: "S's room",
 | 
			
		||||
                  entity: "sensor.temperature_stefan",
 | 
			
		||||
                },
 | 
			
		||||
              ],
 | 
			
		||||
              type: "horizontal-stack",
 | 
			
		||||
              graph: "line",
 | 
			
		||||
              type: "sensor",
 | 
			
		||||
              entity: "sensor.temperature_bedroom",
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
              cards: [
 | 
			
		||||
                {
 | 
			
		||||
                  graph: "line",
 | 
			
		||||
                  type: "sensor",
 | 
			
		||||
                  entity: "sensor.temperature_passage",
 | 
			
		||||
                },
 | 
			
		||||
                {
 | 
			
		||||
                  graph: "line",
 | 
			
		||||
                  type: "sensor",
 | 
			
		||||
                  name: "Bathroom",
 | 
			
		||||
                  entity: "sensor.temperature_downstairs_bathroom",
 | 
			
		||||
                },
 | 
			
		||||
              ],
 | 
			
		||||
              type: "horizontal-stack",
 | 
			
		||||
              graph: "line",
 | 
			
		||||
              type: "sensor",
 | 
			
		||||
              name: "S's room",
 | 
			
		||||
              entity: "sensor.temperature_stefan",
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
              cards: [
 | 
			
		||||
                {
 | 
			
		||||
                  graph: "line",
 | 
			
		||||
                  type: "sensor",
 | 
			
		||||
                  entity: "sensor.temperature_storage",
 | 
			
		||||
                },
 | 
			
		||||
                {
 | 
			
		||||
                  graph: "line",
 | 
			
		||||
                  type: "sensor",
 | 
			
		||||
                  name: "Refrigerator",
 | 
			
		||||
                  entity: "sensor.refrigerator",
 | 
			
		||||
                },
 | 
			
		||||
              ],
 | 
			
		||||
              type: "horizontal-stack",
 | 
			
		||||
              graph: "line",
 | 
			
		||||
              type: "sensor",
 | 
			
		||||
              entity: "sensor.temperature_passage",
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
              graph: "line",
 | 
			
		||||
              type: "sensor",
 | 
			
		||||
              name: "Bathroom",
 | 
			
		||||
              entity: "sensor.temperature_downstairs_bathroom",
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
              graph: "line",
 | 
			
		||||
              type: "sensor",
 | 
			
		||||
              entity: "sensor.temperature_storage",
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
              graph: "line",
 | 
			
		||||
              type: "sensor",
 | 
			
		||||
              name: "Refrigerator",
 | 
			
		||||
              entity: "sensor.refrigerator",
 | 
			
		||||
            },
 | 
			
		||||
          ],
 | 
			
		||||
          type: "vertical-stack",
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          entities: [
 | 
			
		||||
@@ -808,67 +795,6 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
          ],
 | 
			
		||||
          type: "vertical-stack",
 | 
			
		||||
        },
 | 
			
		||||
        // {
 | 
			
		||||
        //   cards: [
 | 
			
		||||
        //     {
 | 
			
		||||
        //       entities: [
 | 
			
		||||
        //         {
 | 
			
		||||
        //           hide_when_off: true,
 | 
			
		||||
        //           toggle: true,
 | 
			
		||||
        //           type: "custom:slider-entity-row",
 | 
			
		||||
        //           name: "Bedside",
 | 
			
		||||
        //           entity: "light.bedside_lamp",
 | 
			
		||||
        //         },
 | 
			
		||||
        //         {
 | 
			
		||||
        //           hide_when_off: true,
 | 
			
		||||
        //           toggle: true,
 | 
			
		||||
        //           type: "custom:slider-entity-row",
 | 
			
		||||
        //           name: "Bedroom",
 | 
			
		||||
        //           entity: "light.bedroom_ceiling_light",
 | 
			
		||||
        //         },
 | 
			
		||||
        //         {
 | 
			
		||||
        //           hide_when_off: true,
 | 
			
		||||
        //           toggle: true,
 | 
			
		||||
        //           type: "custom:slider-entity-row",
 | 
			
		||||
        //           name: "Isa",
 | 
			
		||||
        //           entity: "light.isa_ceiling_light",
 | 
			
		||||
        //         },
 | 
			
		||||
        //         {
 | 
			
		||||
        //           hide_when_off: true,
 | 
			
		||||
        //           toggle: true,
 | 
			
		||||
        //           type: "custom:slider-entity-row",
 | 
			
		||||
        //           name: "Upstairs hallway",
 | 
			
		||||
        //           entity: "light.upstairs_hallway_ceiling_light_level",
 | 
			
		||||
        //         },
 | 
			
		||||
        //         {
 | 
			
		||||
        //           hide_when_off: true,
 | 
			
		||||
        //           toggle: true,
 | 
			
		||||
        //           type: "custom:slider-entity-row",
 | 
			
		||||
        //           name: "Nightlight",
 | 
			
		||||
        //           entity: "light.gateway_light_34ce008bfc4b",
 | 
			
		||||
        //         },
 | 
			
		||||
        //         {
 | 
			
		||||
        //           hide_when_off: true,
 | 
			
		||||
        //           toggle: true,
 | 
			
		||||
        //           type: "custom:slider-entity-row",
 | 
			
		||||
        //           name: "Walk in closet",
 | 
			
		||||
        //           entity: "light.walk_in_closet_lights",
 | 
			
		||||
        //         },
 | 
			
		||||
        //         {
 | 
			
		||||
        //           hide_when_off: true,
 | 
			
		||||
        //           toggle: false,
 | 
			
		||||
        //           type: "custom:slider-entity-row",
 | 
			
		||||
        //           name: "Stefan",
 | 
			
		||||
        //           entity: "light.stefan_lightstrip",
 | 
			
		||||
        //         },
 | 
			
		||||
        //       ],
 | 
			
		||||
        //       show_header_toggle: false,
 | 
			
		||||
        //       type: "entities",
 | 
			
		||||
        //       title: "Upstairs",
 | 
			
		||||
        //     },
 | 
			
		||||
        //   ],
 | 
			
		||||
        //   type: "vertical-stack",
 | 
			
		||||
        // },
 | 
			
		||||
      ],
 | 
			
		||||
      path: "lights",
 | 
			
		||||
      title: "Lights",
 | 
			
		||||
@@ -918,10 +844,6 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
                  name: "Dafang",
 | 
			
		||||
                  icon: "mdi:webcam",
 | 
			
		||||
                },
 | 
			
		||||
                {
 | 
			
		||||
                  name: "IR Hallway",
 | 
			
		||||
                  entity: "sensor.system_ir_blaster",
 | 
			
		||||
                },
 | 
			
		||||
                {
 | 
			
		||||
                  name: "IR Bedroom",
 | 
			
		||||
                  entity: "sensor.system_ir_blaster_bedroom",
 | 
			
		||||
@@ -940,7 +862,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({
 | 
			
		||||
                "sensor.system_ring_chime",
 | 
			
		||||
              ],
 | 
			
		||||
              type: "glance",
 | 
			
		||||
              columns: 5,
 | 
			
		||||
              columns: 4,
 | 
			
		||||
              show_state: false,
 | 
			
		||||
            },
 | 
			
		||||
            {
 | 
			
		||||
 
 | 
			
		||||
@@ -9,5 +9,5 @@ export interface DemoConfig {
 | 
			
		||||
  authorUrl: string;
 | 
			
		||||
  lovelace: (localize: LocalizeFunc) => LovelaceConfig;
 | 
			
		||||
  entities: (localize: LocalizeFunc) => Entity[];
 | 
			
		||||
  theme: () => { [key: string]: string } | null;
 | 
			
		||||
  theme: () => Record<string, string> | null;
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
/* eslint-disable */
 | 
			
		||||
import { LitElement } from "lit-element";
 | 
			
		||||
import { LitElement } from "lit";
 | 
			
		||||
import "./card-tools";
 | 
			
		||||
 | 
			
		||||
class CardModder extends LitElement {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
/* eslint-disable */
 | 
			
		||||
import { html, LitElement } from "lit-element";
 | 
			
		||||
import { html, LitElement } from "lit";
 | 
			
		||||
 | 
			
		||||
if (!window.cardTools) {
 | 
			
		||||
  const version = 0.2;
 | 
			
		||||
 
 | 
			
		||||
@@ -1,12 +1,5 @@
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, state } from "lit/decorators";
 | 
			
		||||
import { CastManager } from "../../../src/cast/cast_manager";
 | 
			
		||||
import { castSendShowDemo } from "../../../src/cast/receiver_messages";
 | 
			
		||||
import "../../../src/components/ha-icon";
 | 
			
		||||
@@ -20,7 +13,7 @@ import { HomeAssistant } from "../../../src/types";
 | 
			
		||||
class CastDemoRow extends LitElement implements LovelaceRow {
 | 
			
		||||
  public hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _castManager?: CastManager | null;
 | 
			
		||||
  @state() private _castManager?: CastManager | null;
 | 
			
		||||
 | 
			
		||||
  public setConfig(_config: CastConfig): void {
 | 
			
		||||
    // No config possible.
 | 
			
		||||
@@ -73,7 +66,7 @@ class CastDemoRow extends LitElement implements LovelaceRow {
 | 
			
		||||
    this.style.display = this._castManager ? "" : "none";
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return css`
 | 
			
		||||
      :host {
 | 
			
		||||
        display: flex;
 | 
			
		||||
 
 | 
			
		||||
@@ -1,14 +1,7 @@
 | 
			
		||||
import "@material/mwc-button";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { until } from "lit-html/directives/until";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { property, state } from "lit/decorators";
 | 
			
		||||
import { until } from "lit/directives/until";
 | 
			
		||||
import "../../../src/components/ha-card";
 | 
			
		||||
import "../../../src/components/ha-circular-progress";
 | 
			
		||||
import { LovelaceCardConfig } from "../../../src/data/lovelace";
 | 
			
		||||
@@ -26,7 +19,7 @@ export class HADemoCard extends LitElement implements LovelaceCard {
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public hass!: MockHomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _switching?: boolean;
 | 
			
		||||
  @state() private _switching?: boolean;
 | 
			
		||||
 | 
			
		||||
  private _hidden = localStorage.hide_demo_card;
 | 
			
		||||
 | 
			
		||||
@@ -113,7 +106,7 @@ export class HADemoCard extends LitElement implements LovelaceCard {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult[] {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return [
 | 
			
		||||
      css`
 | 
			
		||||
        a {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,13 +1,11 @@
 | 
			
		||||
import "../../src/resources/safari-14-attachshadow-patch";
 | 
			
		||||
import "@polymer/polymer/lib/elements/dom-if";
 | 
			
		||||
import "@polymer/polymer/lib/elements/dom-repeat";
 | 
			
		||||
import "../../src/resources/ha-style";
 | 
			
		||||
import "../../src/resources/roboto";
 | 
			
		||||
import "../../src/resources/safari-14-attachshadow-patch";
 | 
			
		||||
import "./ha-demo";
 | 
			
		||||
 | 
			
		||||
/* polyfill for paper-dropdown */
 | 
			
		||||
setTimeout(() => {
 | 
			
		||||
  import(
 | 
			
		||||
    /* webpackChunkName: "polyfill-web-animations-next" */ "web-animations-js/web-animations-next-lite.min"
 | 
			
		||||
  );
 | 
			
		||||
  import("web-animations-js/web-animations-next-lite.min");
 | 
			
		||||
}, 1000);
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,4 @@
 | 
			
		||||
// Compat needs to be first import
 | 
			
		||||
import "../../src/resources/compatibility";
 | 
			
		||||
import { isNavigationClick } from "../../src/common/dom/is-navigation-click";
 | 
			
		||||
import { navigate } from "../../src/common/navigate";
 | 
			
		||||
@@ -21,9 +22,9 @@ import { mockTemplate } from "./stubs/template";
 | 
			
		||||
import { mockTranslations } from "./stubs/translations";
 | 
			
		||||
 | 
			
		||||
class HaDemo extends HomeAssistantAppEl {
 | 
			
		||||
  protected async _initialize() {
 | 
			
		||||
  protected async _initializeHass() {
 | 
			
		||||
    const initial: Partial<MockHomeAssistant> = {
 | 
			
		||||
      panelUrl: (this as any).panelUrl,
 | 
			
		||||
      panelUrl: (this as any)._panelUrl,
 | 
			
		||||
      // Override updateHass so that the correct hass lifecycle methods are called
 | 
			
		||||
      updateHass: (hassUpdate: Partial<HomeAssistant>) =>
 | 
			
		||||
        this._updateHass(hassUpdate),
 | 
			
		||||
@@ -69,7 +70,7 @@ class HaDemo extends HomeAssistantAppEl {
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        e.preventDefault();
 | 
			
		||||
        navigate(this, href);
 | 
			
		||||
        navigate(href);
 | 
			
		||||
      },
 | 
			
		||||
      { capture: true }
 | 
			
		||||
    );
 | 
			
		||||
 
 | 
			
		||||
@@ -3,8 +3,6 @@ import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
 | 
			
		||||
export const mockTranslations = (hass: MockHomeAssistant) => {
 | 
			
		||||
  hass.mockWS(
 | 
			
		||||
    "frontend/get_translations",
 | 
			
		||||
    (/* msg: {language: string, category: string} */) => {
 | 
			
		||||
      return { resources: {} };
 | 
			
		||||
    }
 | 
			
		||||
    (/* msg: {language: string, category: string} */) => ({ resources: {} })
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,7 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
import { safeLoad } from "js-yaml";
 | 
			
		||||
import { load } from "js-yaml";
 | 
			
		||||
import { createCardElement } from "../../../src/panels/lovelace/create-element/create-card-element";
 | 
			
		||||
 | 
			
		||||
class DemoCard extends PolymerElement {
 | 
			
		||||
@@ -15,25 +15,35 @@ class DemoCard extends PolymerElement {
 | 
			
		||||
          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: 16px;
 | 
			
		||||
          margin: 0 16px;
 | 
			
		||||
          overflow: auto;
 | 
			
		||||
          color: var(--primary-text-color);
 | 
			
		||||
        }
 | 
			
		||||
        @media only screen and (max-width: 800px) {
 | 
			
		||||
          .root {
 | 
			
		||||
            flex-direction: column;
 | 
			
		||||
          }
 | 
			
		||||
          pre {
 | 
			
		||||
            margin-left: 0;
 | 
			
		||||
            margin: 16px 0;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      </style>
 | 
			
		||||
      <h2>[[config.heading]]</h2>
 | 
			
		||||
      <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]]">
 | 
			
		||||
@@ -54,6 +64,9 @@ class DemoCard extends PolymerElement {
 | 
			
		||||
        observer: "_configChanged",
 | 
			
		||||
      },
 | 
			
		||||
      showConfig: Boolean,
 | 
			
		||||
      _size: {
 | 
			
		||||
        type: Number,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -67,8 +80,19 @@ class DemoCard extends PolymerElement {
 | 
			
		||||
      card.removeChild(card.lastChild);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const el = this._createCardElement(safeLoad(config.config)[0]);
 | 
			
		||||
    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) {
 | 
			
		||||
 
 | 
			
		||||
@@ -2,10 +2,10 @@ 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 "../../../src/components/ha-switch";
 | 
			
		||||
import "../../../src/components/ha-formfield";
 | 
			
		||||
import "./demo-card";
 | 
			
		||||
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() {
 | 
			
		||||
 
 | 
			
		||||
@@ -2,58 +2,62 @@ 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/state-summary/state-card-content";
 | 
			
		||||
import "../../../src/dialogs/more-info/more-info-content";
 | 
			
		||||
import "../../../src/state-summary/state-card-content";
 | 
			
		||||
 | 
			
		||||
class DemoMoreInfo extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html`
 | 
			
		||||
      <style>
 | 
			
		||||
        :host {
 | 
			
		||||
        .root {
 | 
			
		||||
          display: flex;
 | 
			
		||||
          align-items: start;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #card {
 | 
			
		||||
          max-width: 400px;
 | 
			
		||||
          width: 100vw;
 | 
			
		||||
        }
 | 
			
		||||
        ha-card {
 | 
			
		||||
          width: 333px;
 | 
			
		||||
          width: 352px;
 | 
			
		||||
          padding: 20px 24px;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        state-card-content {
 | 
			
		||||
          display: block;
 | 
			
		||||
          margin-bottom: 16px;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        pre {
 | 
			
		||||
          width: 400px;
 | 
			
		||||
          margin: 16px;
 | 
			
		||||
          margin: 0 16px;
 | 
			
		||||
          overflow: auto;
 | 
			
		||||
          color: var(--primary-text-color);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        @media only screen and (max-width: 800px) {
 | 
			
		||||
          :host {
 | 
			
		||||
          .root {
 | 
			
		||||
            flex-direction: column;
 | 
			
		||||
          }
 | 
			
		||||
          pre {
 | 
			
		||||
            margin-left: 0;
 | 
			
		||||
            margin: 16px 0;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      </style>
 | 
			
		||||
      <ha-card>
 | 
			
		||||
        <state-card-content
 | 
			
		||||
          state-obj="[[_stateObj]]"
 | 
			
		||||
          hass="[[hass]]"
 | 
			
		||||
          in-dialog
 | 
			
		||||
        ></state-card-content>
 | 
			
		||||
      <div class="root">
 | 
			
		||||
        <div id="card">
 | 
			
		||||
          <ha-card>
 | 
			
		||||
            <state-card-content
 | 
			
		||||
              state-obj="[[_stateObj]]"
 | 
			
		||||
              hass="[[hass]]"
 | 
			
		||||
              in-dialog
 | 
			
		||||
            ></state-card-content>
 | 
			
		||||
 | 
			
		||||
        <more-info-content
 | 
			
		||||
          hass="[[hass]]"
 | 
			
		||||
          state-obj="[[_stateObj]]"
 | 
			
		||||
        ></more-info-content>
 | 
			
		||||
      </ha-card>
 | 
			
		||||
      <template is="dom-if" if="[[showConfig]]">
 | 
			
		||||
        <pre>[[_jsonEntity(_stateObj)]]</pre>
 | 
			
		||||
      </template>
 | 
			
		||||
            <more-info-content
 | 
			
		||||
              hass="[[hass]]"
 | 
			
		||||
              state-obj="[[_stateObj]]"
 | 
			
		||||
            ></more-info-content>
 | 
			
		||||
          </ha-card>
 | 
			
		||||
        </div>
 | 
			
		||||
        <template is="dom-if" if="[[showConfig]]">
 | 
			
		||||
          <pre>[[_jsonEntity(_stateObj)]]</pre>
 | 
			
		||||
        </template>
 | 
			
		||||
      </div>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -2,6 +2,8 @@ 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-more-info";
 | 
			
		||||
 | 
			
		||||
@@ -9,6 +11,10 @@ class DemoMoreInfos 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;
 | 
			
		||||
@@ -23,20 +29,31 @@ class DemoMoreInfos extends PolymerElement {
 | 
			
		||||
        .filters {
 | 
			
		||||
          margin-left: 60px;
 | 
			
		||||
        }
 | 
			
		||||
        ha-formfield {
 | 
			
		||||
          margin-right: 16px;
 | 
			
		||||
        }
 | 
			
		||||
      </style>
 | 
			
		||||
      <app-toolbar>
 | 
			
		||||
        <div class="filters">
 | 
			
		||||
          <ha-switch checked="{{_showConfig}}">Show entity</ha-switch>
 | 
			
		||||
          <ha-formfield label="Show entities">
 | 
			
		||||
            <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 class="cards">
 | 
			
		||||
        <template is="dom-repeat" items="[[entities]]">
 | 
			
		||||
          <demo-more-info
 | 
			
		||||
            entity-id="[[item]]"
 | 
			
		||||
            show-config="[[_showConfig]]"
 | 
			
		||||
            hass="[[hass]]"
 | 
			
		||||
          ></demo-more-info>
 | 
			
		||||
        </template>
 | 
			
		||||
      <div id="container">
 | 
			
		||||
        <div class="cards">
 | 
			
		||||
          <template is="dom-repeat" items="[[entities]]">
 | 
			
		||||
            <demo-more-info
 | 
			
		||||
              entity-id="[[item]]"
 | 
			
		||||
              show-config="[[_showConfig]]"
 | 
			
		||||
              hass="[[hass]]"
 | 
			
		||||
            ></demo-more-info>
 | 
			
		||||
          </template>
 | 
			
		||||
        </div>
 | 
			
		||||
      </div>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
@@ -51,6 +68,16 @@ class DemoMoreInfos extends PolymerElement {
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  _showConfigToggled(ev) {
 | 
			
		||||
    this._showConfig = ev.target.checked;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  _darkThemeToggled(ev) {
 | 
			
		||||
    applyThemesOnElement(this.$.container, { themes: {} }, "default", {
 | 
			
		||||
      dark: ev.target.checked,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-more-infos", DemoMoreInfos);
 | 
			
		||||
 
 | 
			
		||||
@@ -7,8 +7,8 @@ export const createMediaPlayerEntities = () => [
 | 
			
		||||
    media_title: "I Wanna Be A Hippy (Flamman & Abraxas Radio Mix)",
 | 
			
		||||
    media_artist: "Technohead",
 | 
			
		||||
    // Pause + Seek + Volume Set + Volume Mute + Previous Track + Next Track + Play Media +
 | 
			
		||||
    // Select Source + Stop + Clear + Play + Shuffle Set + Browse Media
 | 
			
		||||
    supported_features: 195135,
 | 
			
		||||
    // Select Source + Stop + Clear + Play + Shuffle Set
 | 
			
		||||
    supported_features: 64063,
 | 
			
		||||
    entity_picture: "/images/album_cover_2.jpg",
 | 
			
		||||
    media_duration: 300,
 | 
			
		||||
    media_position: 50,
 | 
			
		||||
@@ -24,8 +24,8 @@ export const createMediaPlayerEntities = () => [
 | 
			
		||||
    media_title: "I Wanna Be A Hippy (Flamman & Abraxas Radio Mix)",
 | 
			
		||||
    media_artist: "Technohead",
 | 
			
		||||
    // Pause + Seek + Volume Set + Volume Mute + Previous Track + Next Track + Play Media +
 | 
			
		||||
    // Select Source + Stop + Clear + Play + Shuffle Set
 | 
			
		||||
    supported_features: 64063,
 | 
			
		||||
    // Select Source + Stop + Clear + Play + Shuffle Set + Browse Media
 | 
			
		||||
    supported_features: 195135,
 | 
			
		||||
    entity_picture: "/images/album_cover.jpg",
 | 
			
		||||
    media_duration: 300,
 | 
			
		||||
    media_position: 0,
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										72
									
								
								gallery/src/data/plants.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								gallery/src/data/plants.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,72 @@
 | 
			
		||||
import { getEntity } from "../../../src/fake_data/entity";
 | 
			
		||||
 | 
			
		||||
export const createPlantEntities = () => [
 | 
			
		||||
  getEntity("plant", "lemon_tree", "ok", {
 | 
			
		||||
    problem: "none",
 | 
			
		||||
    sensors: {
 | 
			
		||||
      moisture: "sensor.lemon_tree_moisture",
 | 
			
		||||
      battery: "sensor.lemon_tree_battery",
 | 
			
		||||
      temperature: "sensor.lemon_tree_temperature",
 | 
			
		||||
      conductivity: "sensor.lemon_tree_conductivity",
 | 
			
		||||
      brightness: "sensor.lemon_tree_brightness",
 | 
			
		||||
    },
 | 
			
		||||
    unit_of_measurement_dict: {
 | 
			
		||||
      temperature: "°C",
 | 
			
		||||
      moisture: "%",
 | 
			
		||||
      brightness: "lx",
 | 
			
		||||
      battery: "%",
 | 
			
		||||
      conductivity: "μS/cm",
 | 
			
		||||
    },
 | 
			
		||||
    moisture: 54,
 | 
			
		||||
    battery: 95,
 | 
			
		||||
    temperature: 15.6,
 | 
			
		||||
    conductivity: 1,
 | 
			
		||||
    brightness: 12,
 | 
			
		||||
    max_brightness: 20,
 | 
			
		||||
    friendly_name: "Lemon Tree",
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("plant", "apple_tree", "ok", {
 | 
			
		||||
    problem: "brightness",
 | 
			
		||||
    sensors: {
 | 
			
		||||
      moisture: "sensor.apple_tree_moisture",
 | 
			
		||||
      battery: "sensor.apple_tree_battery",
 | 
			
		||||
      temperature: "sensor.apple_tree_temperature",
 | 
			
		||||
      conductivity: "sensor.apple_tree_conductivity",
 | 
			
		||||
      brightness: "sensor.apple_tree_brightness",
 | 
			
		||||
    },
 | 
			
		||||
    unit_of_measurement_dict: {
 | 
			
		||||
      temperature: "°C",
 | 
			
		||||
      moisture: "%",
 | 
			
		||||
      brightness: "lx",
 | 
			
		||||
      battery: "%",
 | 
			
		||||
      conductivity: "μS/cm",
 | 
			
		||||
    },
 | 
			
		||||
    moisture: 54,
 | 
			
		||||
    battery: 2,
 | 
			
		||||
    temperature: 15.6,
 | 
			
		||||
    conductivity: 1,
 | 
			
		||||
    brightness: 25,
 | 
			
		||||
    max_brightness: 20,
 | 
			
		||||
    friendly_name: "Apple Tree",
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("plant", "sunflowers", "ok", {
 | 
			
		||||
    problem: "moisture, temperature, conductivity",
 | 
			
		||||
    sensors: {
 | 
			
		||||
      moisture: "sensor.sunflowers_moisture",
 | 
			
		||||
      temperature: "sensor.sunflowers_temperature",
 | 
			
		||||
      conductivity: "sensor.sunflowers_conductivity",
 | 
			
		||||
      brightness: "sensor.sunflowers_brightness",
 | 
			
		||||
    },
 | 
			
		||||
    unit_of_measurement_dict: {
 | 
			
		||||
      temperature: "°C",
 | 
			
		||||
      moisture: "%",
 | 
			
		||||
      brightness: "lx",
 | 
			
		||||
      conductivity: "μS/cm",
 | 
			
		||||
    },
 | 
			
		||||
    moisture: 54,
 | 
			
		||||
    temperature: 15.6,
 | 
			
		||||
    conductivity: 1,
 | 
			
		||||
    brightness: 25,
 | 
			
		||||
    entity_picture: "/images/sunflowers.jpg",
 | 
			
		||||
  }),
 | 
			
		||||
];
 | 
			
		||||
							
								
								
									
										349
									
								
								gallery/src/data/traces/basic_trace.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										349
									
								
								gallery/src/data/traces/basic_trace.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,349 @@
 | 
			
		||||
import { DemoTrace } from "./types";
 | 
			
		||||
 | 
			
		||||
export const basicTrace: DemoTrace = {
 | 
			
		||||
  trace: {
 | 
			
		||||
    last_step: "action/2",
 | 
			
		||||
    run_id: "0",
 | 
			
		||||
    state: "stopped",
 | 
			
		||||
    timestamp: {
 | 
			
		||||
      start: "2021-03-25T04:36:51.223693+00:00",
 | 
			
		||||
      finish: "2021-03-25T04:36:51.266132+00:00",
 | 
			
		||||
    },
 | 
			
		||||
    trigger: "state of input_boolean.toggle_1",
 | 
			
		||||
    domain: "automation",
 | 
			
		||||
    item_id: "1615419646544",
 | 
			
		||||
    trace: {
 | 
			
		||||
      "trigger/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "trigger/0",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.223693+00:00",
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "condition/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "condition/0",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.228243+00:00",
 | 
			
		||||
          changed_variables: {
 | 
			
		||||
            trigger: {
 | 
			
		||||
              platform: "state",
 | 
			
		||||
              entity_id: "input_boolean.toggle_1",
 | 
			
		||||
              from_state: {
 | 
			
		||||
                entity_id: "input_boolean.toggle_1",
 | 
			
		||||
                state: "on",
 | 
			
		||||
                attributes: {
 | 
			
		||||
                  editable: true,
 | 
			
		||||
                  friendly_name: "Toggle 1",
 | 
			
		||||
                },
 | 
			
		||||
                last_changed: "2021-03-24T19:03:59.141440+00:00",
 | 
			
		||||
                last_updated: "2021-03-24T19:03:59.141440+00:00",
 | 
			
		||||
                context: {
 | 
			
		||||
                  id: "5d0918eb379214d07554bdab6a08bcff",
 | 
			
		||||
                  parent_id: null,
 | 
			
		||||
                  user_id: null,
 | 
			
		||||
                },
 | 
			
		||||
              },
 | 
			
		||||
              to_state: {
 | 
			
		||||
                entity_id: "input_boolean.toggle_1",
 | 
			
		||||
                state: "off",
 | 
			
		||||
                attributes: {
 | 
			
		||||
                  editable: true,
 | 
			
		||||
                  friendly_name: "Toggle 1",
 | 
			
		||||
                },
 | 
			
		||||
                last_changed: "2021-03-25T04:36:51.220696+00:00",
 | 
			
		||||
                last_updated: "2021-03-25T04:36:51.220696+00:00",
 | 
			
		||||
                context: {
 | 
			
		||||
                  id: "664d6d261450a9ecea6738e97269a149",
 | 
			
		||||
                  parent_id: null,
 | 
			
		||||
                  user_id: "d1b4e89da01445fa8bc98e39fac477ca",
 | 
			
		||||
                },
 | 
			
		||||
              },
 | 
			
		||||
              for: null,
 | 
			
		||||
              attribute: null,
 | 
			
		||||
              description: "state of input_boolean.toggle_1",
 | 
			
		||||
            },
 | 
			
		||||
          },
 | 
			
		||||
          result: {
 | 
			
		||||
            result: true,
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/0",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.243018+00:00",
 | 
			
		||||
          changed_variables: {
 | 
			
		||||
            trigger: {
 | 
			
		||||
              platform: "state",
 | 
			
		||||
              entity_id: "input_boolean.toggle_1",
 | 
			
		||||
              from_state: {
 | 
			
		||||
                entity_id: "input_boolean.toggle_1",
 | 
			
		||||
                state: "on",
 | 
			
		||||
                attributes: {
 | 
			
		||||
                  editable: true,
 | 
			
		||||
                  friendly_name: "Toggle 1",
 | 
			
		||||
                },
 | 
			
		||||
                last_changed: "2021-03-24T19:03:59.141440+00:00",
 | 
			
		||||
                last_updated: "2021-03-24T19:03:59.141440+00:00",
 | 
			
		||||
                context: {
 | 
			
		||||
                  id: "5d0918eb379214d07554bdab6a08bcff",
 | 
			
		||||
                  parent_id: null,
 | 
			
		||||
                  user_id: null,
 | 
			
		||||
                },
 | 
			
		||||
              },
 | 
			
		||||
              to_state: {
 | 
			
		||||
                entity_id: "input_boolean.toggle_1",
 | 
			
		||||
                state: "off",
 | 
			
		||||
                attributes: {
 | 
			
		||||
                  editable: true,
 | 
			
		||||
                  friendly_name: "Toggle 1",
 | 
			
		||||
                },
 | 
			
		||||
                last_changed: "2021-03-25T04:36:51.220696+00:00",
 | 
			
		||||
                last_updated: "2021-03-25T04:36:51.220696+00:00",
 | 
			
		||||
                context: {
 | 
			
		||||
                  id: "664d6d261450a9ecea6738e97269a149",
 | 
			
		||||
                  parent_id: null,
 | 
			
		||||
                  user_id: "d1b4e89da01445fa8bc98e39fac477ca",
 | 
			
		||||
                },
 | 
			
		||||
              },
 | 
			
		||||
              for: null,
 | 
			
		||||
              attribute: null,
 | 
			
		||||
              description: "state of input_boolean.toggle_1",
 | 
			
		||||
            },
 | 
			
		||||
            context: {
 | 
			
		||||
              id: "6cfcae368e7b3686fad6c59e83ae76c9",
 | 
			
		||||
              parent_id: "664d6d261450a9ecea6738e97269a149",
 | 
			
		||||
              user_id: null,
 | 
			
		||||
            },
 | 
			
		||||
          },
 | 
			
		||||
          result: {
 | 
			
		||||
            params: {
 | 
			
		||||
              domain: "input_boolean",
 | 
			
		||||
              service: "toggle",
 | 
			
		||||
              service_data: {},
 | 
			
		||||
              target: {
 | 
			
		||||
                entity_id: ["input_boolean.toggle_4"],
 | 
			
		||||
              },
 | 
			
		||||
            },
 | 
			
		||||
            running_script: false,
 | 
			
		||||
            limit: 10,
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/1": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/1",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.252406+00:00",
 | 
			
		||||
          result: {
 | 
			
		||||
            choice: 0,
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/1/choose/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/1/choose/0",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.254569+00:00",
 | 
			
		||||
          result: {
 | 
			
		||||
            result: true,
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/1/choose/0/conditions/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/1/choose/0/conditions/0",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.254697+00:00",
 | 
			
		||||
          result: {
 | 
			
		||||
            result: true,
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/1/choose/0/sequence/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/1/choose/0/sequence/0",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.257360+00:00",
 | 
			
		||||
          result: {
 | 
			
		||||
            params: {
 | 
			
		||||
              domain: "input_boolean",
 | 
			
		||||
              service: "toggle",
 | 
			
		||||
              service_data: {},
 | 
			
		||||
              target: {
 | 
			
		||||
                entity_id: ["input_boolean.toggle_2"],
 | 
			
		||||
              },
 | 
			
		||||
            },
 | 
			
		||||
            running_script: false,
 | 
			
		||||
            limit: 10,
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/1/choose/0/sequence/1": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/1/choose/0/sequence/1",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.260658+00:00",
 | 
			
		||||
          result: {
 | 
			
		||||
            params: {
 | 
			
		||||
              domain: "input_boolean",
 | 
			
		||||
              service: "toggle",
 | 
			
		||||
              service_data: {},
 | 
			
		||||
              target: {
 | 
			
		||||
                entity_id: ["input_boolean.toggle_3"],
 | 
			
		||||
              },
 | 
			
		||||
            },
 | 
			
		||||
            running_script: false,
 | 
			
		||||
            limit: 10,
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/2": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/2",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.264159+00:00",
 | 
			
		||||
          result: {
 | 
			
		||||
            params: {
 | 
			
		||||
              domain: "input_boolean",
 | 
			
		||||
              service: "toggle",
 | 
			
		||||
              service_data: {},
 | 
			
		||||
              target: {
 | 
			
		||||
                entity_id: ["input_boolean.toggle_4"],
 | 
			
		||||
              },
 | 
			
		||||
            },
 | 
			
		||||
            running_script: false,
 | 
			
		||||
            limit: 10,
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    config: {
 | 
			
		||||
      id: "1615419646544",
 | 
			
		||||
      alias: "Ensure Party mode",
 | 
			
		||||
      description: "",
 | 
			
		||||
      trigger: [
 | 
			
		||||
        {
 | 
			
		||||
          platform: "state",
 | 
			
		||||
          entity_id: "input_boolean.toggle_1",
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      condition: [
 | 
			
		||||
        {
 | 
			
		||||
          condition: "template",
 | 
			
		||||
          alias: "Test if Paulus is home",
 | 
			
		||||
          value_template: "{{ true }}",
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      action: [
 | 
			
		||||
        {
 | 
			
		||||
          service: "input_boolean.toggle",
 | 
			
		||||
          target: {
 | 
			
		||||
            entity_id: "input_boolean.toggle_4",
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          choose: [
 | 
			
		||||
            {
 | 
			
		||||
              alias: "If toggle 3 is on",
 | 
			
		||||
              conditions: [
 | 
			
		||||
                {
 | 
			
		||||
                  condition: "template",
 | 
			
		||||
                  value_template:
 | 
			
		||||
                    "{{ is_state('input_boolean.toggle_3', 'on') }}",
 | 
			
		||||
                },
 | 
			
		||||
              ],
 | 
			
		||||
              sequence: [
 | 
			
		||||
                {
 | 
			
		||||
                  service: "input_boolean.toggle",
 | 
			
		||||
                  alias: "Toggle 2 while 3 is on",
 | 
			
		||||
                  target: {
 | 
			
		||||
                    entity_id: "input_boolean.toggle_2",
 | 
			
		||||
                  },
 | 
			
		||||
                },
 | 
			
		||||
                {
 | 
			
		||||
                  service: "input_boolean.toggle",
 | 
			
		||||
                  alias: "Toggle 3",
 | 
			
		||||
                  target: {
 | 
			
		||||
                    entity_id: "input_boolean.toggle_3",
 | 
			
		||||
                  },
 | 
			
		||||
                },
 | 
			
		||||
              ],
 | 
			
		||||
            },
 | 
			
		||||
          ],
 | 
			
		||||
          default: [
 | 
			
		||||
            {
 | 
			
		||||
              service: "input_boolean.toggle",
 | 
			
		||||
              alias: "Toggle 2",
 | 
			
		||||
              target: {
 | 
			
		||||
                entity_id: "input_boolean.toggle_2",
 | 
			
		||||
              },
 | 
			
		||||
            },
 | 
			
		||||
          ],
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          service: "input_boolean.toggle",
 | 
			
		||||
          target: {
 | 
			
		||||
            entity_id: "input_boolean.toggle_4",
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      mode: "single",
 | 
			
		||||
    },
 | 
			
		||||
    context: {
 | 
			
		||||
      id: "6cfcae368e7b3686fad6c59e83ae76c9",
 | 
			
		||||
      parent_id: "664d6d261450a9ecea6738e97269a149",
 | 
			
		||||
      user_id: null,
 | 
			
		||||
    },
 | 
			
		||||
    script_execution: "finished",
 | 
			
		||||
  },
 | 
			
		||||
  logbookEntries: [
 | 
			
		||||
    {
 | 
			
		||||
      name: "Ensure Party mode",
 | 
			
		||||
      message: "has been triggered by state of input_boolean.toggle_1",
 | 
			
		||||
      source: "state of input_boolean.toggle_1",
 | 
			
		||||
      entity_id: "automation.toggle_toggles",
 | 
			
		||||
      context_id: "6cfcae368e7b3686fad6c59e83ae76c9",
 | 
			
		||||
      when: "2021-03-25T04:36:51.240832+00:00",
 | 
			
		||||
      domain: "automation",
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      when: "2021-03-25T04:36:51.249828+00:00",
 | 
			
		||||
      name: "Toggle 4",
 | 
			
		||||
      state: "on",
 | 
			
		||||
      entity_id: "input_boolean.toggle_4",
 | 
			
		||||
      context_entity_id: "automation.toggle_toggles",
 | 
			
		||||
      context_entity_id_name: "Ensure Party mode",
 | 
			
		||||
      context_event_type: "automation_triggered",
 | 
			
		||||
      context_domain: "automation",
 | 
			
		||||
      context_name: "Ensure Party mode",
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      when: "2021-03-25T04:36:51.258947+00:00",
 | 
			
		||||
      name: "Toggle 2",
 | 
			
		||||
      state: "on",
 | 
			
		||||
      entity_id: "input_boolean.toggle_2",
 | 
			
		||||
      context_entity_id: "automation.toggle_toggles",
 | 
			
		||||
      context_entity_id_name: "Ensure Party mode",
 | 
			
		||||
      context_event_type: "automation_triggered",
 | 
			
		||||
      context_domain: "automation",
 | 
			
		||||
      context_name: "Ensure Party mode",
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      when: "2021-03-25T04:36:51.261806+00:00",
 | 
			
		||||
      name: "Toggle 3",
 | 
			
		||||
      state: "off",
 | 
			
		||||
      entity_id: "input_boolean.toggle_3",
 | 
			
		||||
      context_entity_id: "automation.toggle_toggles",
 | 
			
		||||
      context_entity_id_name: "Ensure Party mode",
 | 
			
		||||
      context_event_type: "automation_triggered",
 | 
			
		||||
      context_domain: "automation",
 | 
			
		||||
      context_name: "Ensure Party mode",
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      when: "2021-03-25T04:36:51.265246+00:00",
 | 
			
		||||
      name: "Toggle 4",
 | 
			
		||||
      state: "off",
 | 
			
		||||
      entity_id: "input_boolean.toggle_4",
 | 
			
		||||
      context_entity_id: "automation.toggle_toggles",
 | 
			
		||||
      context_entity_id_name: "Ensure Party mode",
 | 
			
		||||
      context_event_type: "automation_triggered",
 | 
			
		||||
      context_domain: "automation",
 | 
			
		||||
      context_name: "Ensure Party mode",
 | 
			
		||||
    },
 | 
			
		||||
  ],
 | 
			
		||||
};
 | 
			
		||||
							
								
								
									
										44
									
								
								gallery/src/data/traces/mock-demo-trace.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								gallery/src/data/traces/mock-demo-trace.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,44 @@
 | 
			
		||||
import { LogbookEntry } from "../../../../src/data/logbook";
 | 
			
		||||
import { AutomationTraceExtended } from "../../../../src/data/trace";
 | 
			
		||||
import { DemoTrace } from "./types";
 | 
			
		||||
 | 
			
		||||
export const mockDemoTrace = (
 | 
			
		||||
  tracePartial: Partial<AutomationTraceExtended>,
 | 
			
		||||
  logbookEntries?: LogbookEntry[]
 | 
			
		||||
): DemoTrace => ({
 | 
			
		||||
  trace: {
 | 
			
		||||
    last_step: "",
 | 
			
		||||
    run_id: "0",
 | 
			
		||||
    state: "stopped",
 | 
			
		||||
    timestamp: {
 | 
			
		||||
      start: "2021-03-25T04:36:51.223693+00:00",
 | 
			
		||||
      finish: "2021-03-25T04:36:51.266132+00:00",
 | 
			
		||||
    },
 | 
			
		||||
    trigger: "mocked trigger",
 | 
			
		||||
    domain: "automation",
 | 
			
		||||
    item_id: "1615419646544",
 | 
			
		||||
    trace: {
 | 
			
		||||
      "trigger/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "trigger/0",
 | 
			
		||||
          changed_variables: {
 | 
			
		||||
            trigger: {
 | 
			
		||||
              description: "mocked trigger",
 | 
			
		||||
            },
 | 
			
		||||
          },
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.223693+00:00",
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
    },
 | 
			
		||||
    config: {
 | 
			
		||||
      trigger: [],
 | 
			
		||||
      action: [],
 | 
			
		||||
    },
 | 
			
		||||
    context: {
 | 
			
		||||
      id: "abcd",
 | 
			
		||||
    },
 | 
			
		||||
    script_execution: "finished",
 | 
			
		||||
    ...tracePartial,
 | 
			
		||||
  },
 | 
			
		||||
  logbookEntries: logbookEntries || [],
 | 
			
		||||
});
 | 
			
		||||
							
								
								
									
										214
									
								
								gallery/src/data/traces/motion-light-trace.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										214
									
								
								gallery/src/data/traces/motion-light-trace.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,214 @@
 | 
			
		||||
import { DemoTrace } from "./types";
 | 
			
		||||
 | 
			
		||||
export const motionLightTrace: DemoTrace = {
 | 
			
		||||
  trace: {
 | 
			
		||||
    last_step: "action/3",
 | 
			
		||||
    run_id: "1",
 | 
			
		||||
    state: "stopped",
 | 
			
		||||
    timestamp: {
 | 
			
		||||
      start: "2021-03-14T06:07:01.768006+00:00",
 | 
			
		||||
      finish: "2021-03-14T06:07:53.287525+00:00",
 | 
			
		||||
    },
 | 
			
		||||
    trigger: "state of binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
    domain: "automation",
 | 
			
		||||
    item_id: "1614732497392",
 | 
			
		||||
    trace: {
 | 
			
		||||
      "trigger/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "trigger/0",
 | 
			
		||||
          timestamp: "2021-03-25T04:36:51.223693+00:00",
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/0": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/0",
 | 
			
		||||
          timestamp: "2021-03-14T06:07:01.771038+00:00",
 | 
			
		||||
          changed_variables: {
 | 
			
		||||
            trigger: {
 | 
			
		||||
              platform: "state",
 | 
			
		||||
              entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
              from_state: {
 | 
			
		||||
                entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
                state: "off",
 | 
			
		||||
                attributes: {
 | 
			
		||||
                  friendly_name: "Paulus’s MacBook Pro Camera In Use",
 | 
			
		||||
                  icon: "mdi:camera-off",
 | 
			
		||||
                },
 | 
			
		||||
                last_changed: "2021-03-14T06:06:29.235325+00:00",
 | 
			
		||||
                last_updated: "2021-03-14T06:06:29.235325+00:00",
 | 
			
		||||
                context: {
 | 
			
		||||
                  id: "ad4864c5ce957c38a07b50378eeb245d",
 | 
			
		||||
                  parent_id: null,
 | 
			
		||||
                  user_id: null,
 | 
			
		||||
                },
 | 
			
		||||
              },
 | 
			
		||||
              to_state: {
 | 
			
		||||
                entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
                state: "on",
 | 
			
		||||
                attributes: {
 | 
			
		||||
                  friendly_name: "Paulus’s MacBook Pro Camera In Use",
 | 
			
		||||
                  icon: "mdi:camera",
 | 
			
		||||
                },
 | 
			
		||||
                last_changed: "2021-03-14T06:07:01.762009+00:00",
 | 
			
		||||
                last_updated: "2021-03-14T06:07:01.762009+00:00",
 | 
			
		||||
                context: {
 | 
			
		||||
                  id: "e22ddfd5f11dc4aad9a52fc10dab613b",
 | 
			
		||||
                  parent_id: null,
 | 
			
		||||
                  user_id: null,
 | 
			
		||||
                },
 | 
			
		||||
              },
 | 
			
		||||
              for: null,
 | 
			
		||||
              attribute: null,
 | 
			
		||||
              description:
 | 
			
		||||
                "state of binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
            },
 | 
			
		||||
            context: {
 | 
			
		||||
              id: "43b6ee9293a551c5cc14e8eb60af54ba",
 | 
			
		||||
              parent_id: "e22ddfd5f11dc4aad9a52fc10dab613b",
 | 
			
		||||
              user_id: null,
 | 
			
		||||
            },
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/1": [
 | 
			
		||||
        { path: "action/1", timestamp: "2021-03-14T06:07:01.875316+00:00" },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/2": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/2",
 | 
			
		||||
          timestamp: "2021-03-14T06:07:53.195013+00:00",
 | 
			
		||||
          changed_variables: {
 | 
			
		||||
            wait: {
 | 
			
		||||
              remaining: null,
 | 
			
		||||
              trigger: {
 | 
			
		||||
                platform: "state",
 | 
			
		||||
                entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
                from_state: {
 | 
			
		||||
                  entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
                  state: "on",
 | 
			
		||||
                  attributes: {
 | 
			
		||||
                    friendly_name: "Paulus’s MacBook Pro Camera In Use",
 | 
			
		||||
                    icon: "mdi:camera",
 | 
			
		||||
                  },
 | 
			
		||||
                  last_changed: "2021-03-14T06:07:01.762009+00:00",
 | 
			
		||||
                  last_updated: "2021-03-14T06:07:01.762009+00:00",
 | 
			
		||||
                  context: {
 | 
			
		||||
                    id: "e22ddfd5f11dc4aad9a52fc10dab613b",
 | 
			
		||||
                    parent_id: null,
 | 
			
		||||
                    user_id: null,
 | 
			
		||||
                  },
 | 
			
		||||
                },
 | 
			
		||||
                to_state: {
 | 
			
		||||
                  entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
                  state: "off",
 | 
			
		||||
                  attributes: {
 | 
			
		||||
                    friendly_name: "Paulus’s MacBook Pro Camera In Use",
 | 
			
		||||
                    icon: "mdi:camera-off",
 | 
			
		||||
                  },
 | 
			
		||||
                  last_changed: "2021-03-14T06:07:53.186755+00:00",
 | 
			
		||||
                  last_updated: "2021-03-14T06:07:53.186755+00:00",
 | 
			
		||||
                  context: {
 | 
			
		||||
                    id: "b2308cc91d509ea8e0c623331ab178d6",
 | 
			
		||||
                    parent_id: null,
 | 
			
		||||
                    user_id: null,
 | 
			
		||||
                  },
 | 
			
		||||
                },
 | 
			
		||||
                for: null,
 | 
			
		||||
                attribute: null,
 | 
			
		||||
                description:
 | 
			
		||||
                  "state of binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
              },
 | 
			
		||||
            },
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      "action/3": [
 | 
			
		||||
        {
 | 
			
		||||
          path: "action/3",
 | 
			
		||||
          timestamp: "2021-03-14T06:07:53.196014+00:00",
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
    },
 | 
			
		||||
    config: {
 | 
			
		||||
      mode: "restart",
 | 
			
		||||
      max_exceeded: "silent",
 | 
			
		||||
      trigger: [
 | 
			
		||||
        {
 | 
			
		||||
          platform: "state",
 | 
			
		||||
          entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
          from: "off",
 | 
			
		||||
          to: "on",
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      action: [
 | 
			
		||||
        {
 | 
			
		||||
          service: "light.turn_on",
 | 
			
		||||
          target: {
 | 
			
		||||
            entity_id: "light.elgato_key_light_air",
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          wait_for_trigger: [
 | 
			
		||||
            {
 | 
			
		||||
              platform: "state",
 | 
			
		||||
              entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
              from: "on",
 | 
			
		||||
              to: "off",
 | 
			
		||||
            },
 | 
			
		||||
          ],
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          delay: 0,
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
          service: "light.turn_off",
 | 
			
		||||
          target: {
 | 
			
		||||
            entity_id: "light.elgato_key_light_air",
 | 
			
		||||
          },
 | 
			
		||||
        },
 | 
			
		||||
      ],
 | 
			
		||||
      id: "1614732497392",
 | 
			
		||||
      alias: "Auto Elgato",
 | 
			
		||||
      description: "",
 | 
			
		||||
    },
 | 
			
		||||
    context: {
 | 
			
		||||
      id: "43b6ee9293a551c5cc14e8eb60af54ba",
 | 
			
		||||
      parent_id: "e22ddfd5f11dc4aad9a52fc10dab613b",
 | 
			
		||||
      user_id: null,
 | 
			
		||||
    },
 | 
			
		||||
    script_execution: "finished",
 | 
			
		||||
  },
 | 
			
		||||
  logbookEntries: [
 | 
			
		||||
    {
 | 
			
		||||
      name: "Auto Elgato",
 | 
			
		||||
      message:
 | 
			
		||||
        "has been triggered by state of binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
      source: "state of binary_sensor.pauluss_macbook_pro_camera_in_use",
 | 
			
		||||
      entity_id: "automation.auto_elgato",
 | 
			
		||||
      when: "2021-03-14T06:07:01.768492+00:00",
 | 
			
		||||
      domain: "automation",
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      when: "2021-03-14T06:07:01.872187+00:00",
 | 
			
		||||
      name: "Elgato Key Light Air",
 | 
			
		||||
      state: "on",
 | 
			
		||||
      entity_id: "light.elgato_key_light_air",
 | 
			
		||||
      context_entity_id: "automation.auto_elgato",
 | 
			
		||||
      context_entity_id_name: "Auto Elgato",
 | 
			
		||||
      context_event_type: "automation_triggered",
 | 
			
		||||
      context_domain: "automation",
 | 
			
		||||
      context_name: "Auto Elgato",
 | 
			
		||||
    },
 | 
			
		||||
    {
 | 
			
		||||
      when: "2021-03-14T06:07:53.284505+00:00",
 | 
			
		||||
      name: "Elgato Key Light Air",
 | 
			
		||||
      state: "off",
 | 
			
		||||
      entity_id: "light.elgato_key_light_air",
 | 
			
		||||
      context_entity_id: "automation.auto_elgato",
 | 
			
		||||
      context_entity_id_name: "Auto Elgato",
 | 
			
		||||
      context_event_type: "automation_triggered",
 | 
			
		||||
      context_domain: "automation",
 | 
			
		||||
      context_name: "Auto Elgato",
 | 
			
		||||
    },
 | 
			
		||||
  ],
 | 
			
		||||
};
 | 
			
		||||
							
								
								
									
										7
									
								
								gallery/src/data/traces/types.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								gallery/src/data/traces/types.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,7 @@
 | 
			
		||||
import { AutomationTraceExtended } from "../../../../src/data/trace";
 | 
			
		||||
import { LogbookEntry } from "../../../../src/data/logbook";
 | 
			
		||||
 | 
			
		||||
export interface DemoTrace {
 | 
			
		||||
  trace: AutomationTraceExtended;
 | 
			
		||||
  logbookEntries: LogbookEntry[];
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										96
									
								
								gallery/src/demos/demo-automation-describe-action.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								gallery/src/demos/demo-automation-describe-action.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,96 @@
 | 
			
		||||
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";
 | 
			
		||||
 | 
			
		||||
const actions = [
 | 
			
		||||
  { wait_template: "{{ true }}", alias: "Something with an alias" },
 | 
			
		||||
  { delay: "0:05" },
 | 
			
		||||
  { wait_template: "{{ true }}" },
 | 
			
		||||
  {
 | 
			
		||||
    condition: "template",
 | 
			
		||||
    value_template: "{{ true }}",
 | 
			
		||||
  },
 | 
			
		||||
  { event: "happy_event" },
 | 
			
		||||
  {
 | 
			
		||||
    device_id: "abcdefgh",
 | 
			
		||||
    domain: "plex",
 | 
			
		||||
    entity_id: "media_player.kitchen",
 | 
			
		||||
  },
 | 
			
		||||
  { scene: "scene.kitchen_morning" },
 | 
			
		||||
  {
 | 
			
		||||
    wait_for_trigger: [
 | 
			
		||||
      {
 | 
			
		||||
        platform: "state",
 | 
			
		||||
        entity_id: "input_boolean.toggle_1",
 | 
			
		||||
      },
 | 
			
		||||
    ],
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    variables: {
 | 
			
		||||
      hello: "world",
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    service: "input_boolean.toggle",
 | 
			
		||||
    target: {
 | 
			
		||||
      entity_id: "input_boolean.toggle_4",
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
@customElement("demo-automation-describe-action")
 | 
			
		||||
export class DemoAutomationDescribeAction extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    if (!this.hass) {
 | 
			
		||||
      return html``;
 | 
			
		||||
    }
 | 
			
		||||
    return html`
 | 
			
		||||
      <ha-card header="Actions">
 | 
			
		||||
        ${actions.map(
 | 
			
		||||
          (conf) => html`
 | 
			
		||||
            <div class="action">
 | 
			
		||||
              <span>${describeAction(this.hass, conf as any)}</span>
 | 
			
		||||
              <pre>${dump(conf)}</pre>
 | 
			
		||||
            </div>
 | 
			
		||||
          `
 | 
			
		||||
        )}
 | 
			
		||||
      </ha-card>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  protected firstUpdated(changedProps) {
 | 
			
		||||
    super.firstUpdated(changedProps);
 | 
			
		||||
    const hass = provideHass(this);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles() {
 | 
			
		||||
    return css`
 | 
			
		||||
      ha-card {
 | 
			
		||||
        max-width: 600px;
 | 
			
		||||
        margin: 24px auto;
 | 
			
		||||
      }
 | 
			
		||||
      .action {
 | 
			
		||||
        padding: 16px;
 | 
			
		||||
        display: flex;
 | 
			
		||||
        align-items: center;
 | 
			
		||||
        justify-content: space-between;
 | 
			
		||||
      }
 | 
			
		||||
      span {
 | 
			
		||||
        margin-right: 16px;
 | 
			
		||||
      }
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-automation-describe-action": DemoAutomationDescribeAction;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										60
									
								
								gallery/src/demos/demo-automation-describe-condition.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								gallery/src/demos/demo-automation-describe-condition.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,60 @@
 | 
			
		||||
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";
 | 
			
		||||
 | 
			
		||||
const conditions = [
 | 
			
		||||
  { condition: "and" },
 | 
			
		||||
  { condition: "not" },
 | 
			
		||||
  { condition: "or" },
 | 
			
		||||
  { condition: "state" },
 | 
			
		||||
  { condition: "numeric_state" },
 | 
			
		||||
  { condition: "sun", after: "sunset" },
 | 
			
		||||
  { condition: "sun", after: "sunrise" },
 | 
			
		||||
  { condition: "zone" },
 | 
			
		||||
  { condition: "time" },
 | 
			
		||||
  { condition: "template" },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
@customElement("demo-automation-describe-condition")
 | 
			
		||||
export class DemoAutomationDescribeCondition extends LitElement {
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`
 | 
			
		||||
      <ha-card header="Conditions">
 | 
			
		||||
        ${conditions.map(
 | 
			
		||||
          (conf) => html`
 | 
			
		||||
            <div class="condition">
 | 
			
		||||
              <span>${describeCondition(conf as any)}</span>
 | 
			
		||||
              <pre>${dump(conf)}</pre>
 | 
			
		||||
            </div>
 | 
			
		||||
          `
 | 
			
		||||
        )}
 | 
			
		||||
      </ha-card>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles() {
 | 
			
		||||
    return css`
 | 
			
		||||
      ha-card {
 | 
			
		||||
        max-width: 600px;
 | 
			
		||||
        margin: 24px auto;
 | 
			
		||||
      }
 | 
			
		||||
      .condition {
 | 
			
		||||
        padding: 16px;
 | 
			
		||||
        display: flex;
 | 
			
		||||
        align-items: center;
 | 
			
		||||
        justify-content: space-between;
 | 
			
		||||
      }
 | 
			
		||||
      span {
 | 
			
		||||
        margin-right: 16px;
 | 
			
		||||
      }
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-automation-describe-condition": DemoAutomationDescribeCondition;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										63
									
								
								gallery/src/demos/demo-automation-describe-trigger.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								gallery/src/demos/demo-automation-describe-trigger.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,63 @@
 | 
			
		||||
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";
 | 
			
		||||
 | 
			
		||||
const triggers = [
 | 
			
		||||
  { platform: "state" },
 | 
			
		||||
  { platform: "mqtt" },
 | 
			
		||||
  { platform: "geo_location" },
 | 
			
		||||
  { platform: "homeassistant" },
 | 
			
		||||
  { platform: "numeric_state" },
 | 
			
		||||
  { platform: "sun" },
 | 
			
		||||
  { platform: "time_pattern" },
 | 
			
		||||
  { platform: "webhook" },
 | 
			
		||||
  { platform: "zone" },
 | 
			
		||||
  { platform: "tag" },
 | 
			
		||||
  { platform: "time" },
 | 
			
		||||
  { platform: "template" },
 | 
			
		||||
  { platform: "event" },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
@customElement("demo-automation-describe-trigger")
 | 
			
		||||
export class DemoAutomationDescribeTrigger extends LitElement {
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`
 | 
			
		||||
      <ha-card header="Triggers">
 | 
			
		||||
        ${triggers.map(
 | 
			
		||||
          (conf) => html`
 | 
			
		||||
            <div class="trigger">
 | 
			
		||||
              <span>${describeTrigger(conf as any)}</span>
 | 
			
		||||
              <pre>${dump(conf)}</pre>
 | 
			
		||||
            </div>
 | 
			
		||||
          `
 | 
			
		||||
        )}
 | 
			
		||||
      </ha-card>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles() {
 | 
			
		||||
    return css`
 | 
			
		||||
      ha-card {
 | 
			
		||||
        max-width: 600px;
 | 
			
		||||
        margin: 24px auto;
 | 
			
		||||
      }
 | 
			
		||||
      .trigger {
 | 
			
		||||
        padding: 16px;
 | 
			
		||||
        display: flex;
 | 
			
		||||
        align-items: center;
 | 
			
		||||
        justify-content: space-between;
 | 
			
		||||
      }
 | 
			
		||||
      span {
 | 
			
		||||
        margin-right: 16px;
 | 
			
		||||
      }
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-automation-describe-trigger": DemoAutomationDescribeTrigger;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										81
									
								
								gallery/src/demos/demo-automation-trace-timeline.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								gallery/src/demos/demo-automation-trace-timeline.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,81 @@
 | 
			
		||||
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";
 | 
			
		||||
 | 
			
		||||
const traces: DemoTrace[] = [
 | 
			
		||||
  mockDemoTrace({ state: "running" }),
 | 
			
		||||
  mockDemoTrace({ state: "debugged" }),
 | 
			
		||||
  mockDemoTrace({ state: "stopped", script_execution: "failed_conditions" }),
 | 
			
		||||
  mockDemoTrace({ state: "stopped", script_execution: "failed_single" }),
 | 
			
		||||
  mockDemoTrace({ state: "stopped", script_execution: "failed_max_runs" }),
 | 
			
		||||
  mockDemoTrace({ state: "stopped", script_execution: "finished" }),
 | 
			
		||||
  mockDemoTrace({ state: "stopped", script_execution: "aborted" }),
 | 
			
		||||
  mockDemoTrace({
 | 
			
		||||
    state: "stopped",
 | 
			
		||||
    script_execution: "error",
 | 
			
		||||
    error: 'Variable "beer" cannot be None',
 | 
			
		||||
  }),
 | 
			
		||||
  mockDemoTrace({ state: "stopped", script_execution: "cancelled" }),
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
@customElement("demo-automation-trace-timeline")
 | 
			
		||||
export class DemoAutomationTraceTimeline extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) hass?: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    if (!this.hass) {
 | 
			
		||||
      return html``;
 | 
			
		||||
    }
 | 
			
		||||
    return html`
 | 
			
		||||
      ${traces.map(
 | 
			
		||||
        (trace) => html`
 | 
			
		||||
          <ha-card .header=${trace.trace.config.alias}>
 | 
			
		||||
            <div class="card-content">
 | 
			
		||||
              <hat-trace-timeline
 | 
			
		||||
                .hass=${this.hass}
 | 
			
		||||
                .trace=${trace.trace}
 | 
			
		||||
                .logbookEntries=${trace.logbookEntries}
 | 
			
		||||
              ></hat-trace-timeline>
 | 
			
		||||
              <button @click=${() => console.log(trace)}>Log trace</button>
 | 
			
		||||
            </div>
 | 
			
		||||
          </ha-card>
 | 
			
		||||
        `
 | 
			
		||||
      )}
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  protected firstUpdated(changedProps) {
 | 
			
		||||
    super.firstUpdated(changedProps);
 | 
			
		||||
    const hass = provideHass(this);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles() {
 | 
			
		||||
    return css`
 | 
			
		||||
      ha-card {
 | 
			
		||||
        max-width: 600px;
 | 
			
		||||
        margin: 24px;
 | 
			
		||||
      }
 | 
			
		||||
      .card-content {
 | 
			
		||||
        display: flex;
 | 
			
		||||
      }
 | 
			
		||||
      button {
 | 
			
		||||
        position: absolute;
 | 
			
		||||
        top: 0;
 | 
			
		||||
        right: 0;
 | 
			
		||||
      }
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-automation-trace-timeline": DemoAutomationTraceTimeline;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										91
									
								
								gallery/src/demos/demo-automation-trace.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								gallery/src/demos/demo-automation-trace.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,91 @@
 | 
			
		||||
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 { 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 { customElement, property, state } from "lit/decorators";
 | 
			
		||||
 | 
			
		||||
const traces: DemoTrace[] = [basicTrace, motionLightTrace];
 | 
			
		||||
 | 
			
		||||
@customElement("demo-automation-trace")
 | 
			
		||||
export class DemoAutomationTrace extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) hass?: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @state() private _selected = {};
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    if (!this.hass) {
 | 
			
		||||
      return html``;
 | 
			
		||||
    }
 | 
			
		||||
    return html`
 | 
			
		||||
      ${traces.map(
 | 
			
		||||
        (trace, idx) => html`
 | 
			
		||||
          <ha-card .header=${trace.trace.config.alias}>
 | 
			
		||||
            <div class="card-content">
 | 
			
		||||
              <hat-script-graph
 | 
			
		||||
                .trace=${trace.trace}
 | 
			
		||||
                .selected=${this._selected[idx]}
 | 
			
		||||
                @graph-node-selected=${(ev) => {
 | 
			
		||||
                  this._selected = { ...this._selected, [idx]: ev.detail.path };
 | 
			
		||||
                }}
 | 
			
		||||
              ></hat-script-graph>
 | 
			
		||||
              <hat-trace-timeline
 | 
			
		||||
                allowPick
 | 
			
		||||
                .hass=${this.hass}
 | 
			
		||||
                .trace=${trace.trace}
 | 
			
		||||
                .logbookEntries=${trace.logbookEntries}
 | 
			
		||||
                .selectedPath=${this._selected[idx]}
 | 
			
		||||
                @value-changed=${(ev) => {
 | 
			
		||||
                  this._selected = {
 | 
			
		||||
                    ...this._selected,
 | 
			
		||||
                    [idx]: ev.detail.value,
 | 
			
		||||
                  };
 | 
			
		||||
                }}
 | 
			
		||||
              ></hat-trace-timeline>
 | 
			
		||||
              <button @click=${() => console.log(trace)}>Log trace</button>
 | 
			
		||||
            </div>
 | 
			
		||||
          </ha-card>
 | 
			
		||||
        `
 | 
			
		||||
      )}
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  protected firstUpdated(changedProps) {
 | 
			
		||||
    super.firstUpdated(changedProps);
 | 
			
		||||
    const hass = provideHass(this);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles() {
 | 
			
		||||
    return css`
 | 
			
		||||
      ha-card {
 | 
			
		||||
        max-width: 600px;
 | 
			
		||||
        margin: 24px;
 | 
			
		||||
      }
 | 
			
		||||
      .card-content {
 | 
			
		||||
        display: flex;
 | 
			
		||||
      }
 | 
			
		||||
      .card-content > * {
 | 
			
		||||
        margin-right: 16px;
 | 
			
		||||
      }
 | 
			
		||||
      .card-content > *:last-child {
 | 
			
		||||
        margin-right: 0;
 | 
			
		||||
      }
 | 
			
		||||
      button {
 | 
			
		||||
        position: absolute;
 | 
			
		||||
        top: 0;
 | 
			
		||||
        right: 0;
 | 
			
		||||
      }
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-automation-trace": DemoAutomationTrace;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -71,37 +70,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoAlarmPanelEntity extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html`
 | 
			
		||||
      <demo-cards
 | 
			
		||||
        id="demos"
 | 
			
		||||
        hass="[[hass]]"
 | 
			
		||||
        configs="[[_configs]]"
 | 
			
		||||
      ></demo-cards>
 | 
			
		||||
    `;
 | 
			
		||||
@customElement("demo-hui-alarm-panel-card")
 | 
			
		||||
class DemoAlarmPanelEntity extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
      hass: Object,
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    this._setupDemo();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _setupDemo() {
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
    await hass.updateTranslations(null, "en");
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-alarm-panel-card", DemoAlarmPanelEntity);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-alarm-panel-card": DemoAlarmPanelEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -53,33 +52,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoConditional extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html`
 | 
			
		||||
      <demo-cards
 | 
			
		||||
        id="demos"
 | 
			
		||||
        hass="[[hass]]"
 | 
			
		||||
        configs="[[_configs]]"
 | 
			
		||||
      ></demo-cards>
 | 
			
		||||
    `;
 | 
			
		||||
@customElement("demo-hui-conditional-card")
 | 
			
		||||
class DemoConditional extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
      hass: Object,
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-conditional-card", DemoConditional);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-conditional-card": DemoConditional;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -217,26 +216,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoEntities extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-entities-card")
 | 
			
		||||
class DemoEntities extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-entities-card", DemoEntities);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-entities-card": DemoEntities;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -20,10 +19,10 @@ const CONFIGS = [
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "With Name",
 | 
			
		||||
    heading: "With Name (defined in card)",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: button
 | 
			
		||||
  name: Bedroom
 | 
			
		||||
  name: Custom Name
 | 
			
		||||
  entity: light.bed_light
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
@@ -32,7 +31,7 @@ const CONFIGS = [
 | 
			
		||||
    config: `
 | 
			
		||||
- type: button
 | 
			
		||||
  entity: light.bed_light
 | 
			
		||||
  icon: mdi:hotel
 | 
			
		||||
  icon: mdi:tools
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
@@ -48,7 +47,7 @@ const CONFIGS = [
 | 
			
		||||
    config: `
 | 
			
		||||
- type: button
 | 
			
		||||
  entity: light.bed_light
 | 
			
		||||
  tap_action: 
 | 
			
		||||
  tap_action:
 | 
			
		||||
    action: toggle
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
@@ -69,33 +68,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoButtonEntity extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html`
 | 
			
		||||
      <demo-cards
 | 
			
		||||
        id="demos"
 | 
			
		||||
        hass="[[hass]]"
 | 
			
		||||
        configs="[[_configs]]"
 | 
			
		||||
      ></demo-cards>
 | 
			
		||||
    `;
 | 
			
		||||
@customElement("demo-hui-entity-button-card")
 | 
			
		||||
class DemoButtonEntity extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
      hass: Object,
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-entity-button-card", DemoButtonEntity);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-entity-button-card": DemoButtonEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -43,7 +42,7 @@ const ENTITIES = [
 | 
			
		||||
 | 
			
		||||
const CONFIGS = [
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Controller",
 | 
			
		||||
    heading: "Unfiltered controller",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: entities
 | 
			
		||||
  entities:
 | 
			
		||||
@@ -53,7 +52,7 @@ const CONFIGS = [
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Basic",
 | 
			
		||||
    heading: "Filtered entities card",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: entity-filter
 | 
			
		||||
  entities:
 | 
			
		||||
@@ -69,7 +68,27 @@ const CONFIGS = [
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "With card config",
 | 
			
		||||
    heading: 'With "entities" card config',
 | 
			
		||||
    config: `
 | 
			
		||||
- type: entity-filter
 | 
			
		||||
  entities:
 | 
			
		||||
    - device_tracker.demo_anne_therese
 | 
			
		||||
    - device_tracker.demo_home_boy
 | 
			
		||||
    - device_tracker.demo_paulus
 | 
			
		||||
    - light.bed_light
 | 
			
		||||
    - light.ceiling_lights
 | 
			
		||||
    - light.kitchen_lights
 | 
			
		||||
  state_filter:
 | 
			
		||||
    - "on"
 | 
			
		||||
    - not_home
 | 
			
		||||
  card:
 | 
			
		||||
    type: entities
 | 
			
		||||
    title: Custom Title
 | 
			
		||||
    show_header_toggle: false
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: 'With "glance" card config',
 | 
			
		||||
    config: `
 | 
			
		||||
- type: entity-filter
 | 
			
		||||
  entities:
 | 
			
		||||
@@ -84,31 +103,31 @@ const CONFIGS = [
 | 
			
		||||
    - not_home
 | 
			
		||||
  card:
 | 
			
		||||
    type: glance
 | 
			
		||||
    show_state: false
 | 
			
		||||
    show_state: true
 | 
			
		||||
    title: Custom Title
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoFilter extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-entity-filter-card")
 | 
			
		||||
class DemoEntityFilter extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-entity-filter-card", DemoFilter);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-entity-filter-card": DemoEntityFilter;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -107,26 +106,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoGaugeEntity extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-gauge-card")
 | 
			
		||||
class DemoGaugeEntity extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-gauge-card", DemoGaugeEntity);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-gauge-card": DemoGaugeEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -77,7 +76,8 @@ const CONFIGS = [
 | 
			
		||||
    heading: "With title",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: glance
 | 
			
		||||
  title: This is glance
 | 
			
		||||
  title: Custom title
 | 
			
		||||
  columns: 4
 | 
			
		||||
  entities:
 | 
			
		||||
    - device_tracker.demo_paulus
 | 
			
		||||
    - media_player.living_room
 | 
			
		||||
@@ -104,9 +104,10 @@ const CONFIGS = [
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "No name",
 | 
			
		||||
    heading: "No entity names",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: glance
 | 
			
		||||
  columns: 4
 | 
			
		||||
  show_name: false
 | 
			
		||||
  entities:
 | 
			
		||||
    - device_tracker.demo_paulus
 | 
			
		||||
@@ -119,9 +120,10 @@ const CONFIGS = [
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "No state",
 | 
			
		||||
    heading: "No state labels",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: glance
 | 
			
		||||
  columns: 4
 | 
			
		||||
  show_state: false
 | 
			
		||||
  entities:
 | 
			
		||||
    - device_tracker.demo_paulus
 | 
			
		||||
@@ -134,9 +136,10 @@ const CONFIGS = [
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "No name and no state",
 | 
			
		||||
    heading: "No names and no state labels",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: glance
 | 
			
		||||
  columns: 4
 | 
			
		||||
  show_name: false
 | 
			
		||||
  show_state: false
 | 
			
		||||
  entities:
 | 
			
		||||
@@ -150,47 +153,24 @@ const CONFIGS = [
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Custom name, custom icon",
 | 
			
		||||
    heading: "Custom name + custom icon",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: glance
 | 
			
		||||
  columns: 4
 | 
			
		||||
  entities:
 | 
			
		||||
    - entity: device_tracker.demo_paulus
 | 
			
		||||
      name: ¯\\_(ツ)_/¯
 | 
			
		||||
      icon: mdi:home-assistant
 | 
			
		||||
    - media_player.living_room
 | 
			
		||||
    - sun.sun
 | 
			
		||||
    - cover.kitchen_window
 | 
			
		||||
    - entity: light.kitchen_lights
 | 
			
		||||
      icon: mdi:alarm-light
 | 
			
		||||
    - lock.kitchen_door
 | 
			
		||||
    - light.ceiling_lights
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Custom tap action",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: glance
 | 
			
		||||
  entities:
 | 
			
		||||
    - entity: lock.kitchen_door
 | 
			
		||||
      tap_action:
 | 
			
		||||
        type: toggle
 | 
			
		||||
    - entity: light.ceiling_lights
 | 
			
		||||
      tap_action:
 | 
			
		||||
        action: call-service
 | 
			
		||||
        service: light.turn_on
 | 
			
		||||
        service_data:
 | 
			
		||||
          entity_id: light.ceiling_lights
 | 
			
		||||
    - device_tracker.demo_paulus
 | 
			
		||||
    - media_player.living_room
 | 
			
		||||
    - sun.sun
 | 
			
		||||
    - cover.kitchen_window
 | 
			
		||||
    - light.kitchen_lights
 | 
			
		||||
    - entity: media_player.living_room
 | 
			
		||||
      name: ¯\\_(ツ)_/¯
 | 
			
		||||
      icon: mdi:home-assistant
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Selectively hidden name",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: glance
 | 
			
		||||
  columns: 4
 | 
			
		||||
  entities:
 | 
			
		||||
    - device_tracker.demo_paulus
 | 
			
		||||
    - entity: media_player.living_room
 | 
			
		||||
@@ -199,45 +179,55 @@ const CONFIGS = [
 | 
			
		||||
    - entity: cover.kitchen_window
 | 
			
		||||
      name:
 | 
			
		||||
    - light.kitchen_lights
 | 
			
		||||
    - entity: lock.kitchen_door
 | 
			
		||||
      name:
 | 
			
		||||
    - light.ceiling_lights
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Primary theme",
 | 
			
		||||
    heading: "Custom tap action",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: glance
 | 
			
		||||
  theming: primary
 | 
			
		||||
  columns: 4
 | 
			
		||||
  entities:
 | 
			
		||||
    - device_tracker.demo_paulus
 | 
			
		||||
    - media_player.living_room
 | 
			
		||||
    - sun.sun
 | 
			
		||||
    - cover.kitchen_window
 | 
			
		||||
    - light.kitchen_lights
 | 
			
		||||
    - lock.kitchen_door
 | 
			
		||||
    - light.ceiling_lights
 | 
			
		||||
    - entity: lock.kitchen_door
 | 
			
		||||
      name: Custom
 | 
			
		||||
      tap_action:
 | 
			
		||||
        type: toggle
 | 
			
		||||
    - entity: light.ceiling_lights
 | 
			
		||||
      name: Custom
 | 
			
		||||
      tap_action:
 | 
			
		||||
        action: call-service
 | 
			
		||||
        service: light.turn_on
 | 
			
		||||
        service_data:
 | 
			
		||||
          entity_id: light.ceiling_lights
 | 
			
		||||
    - entity: sun.sun
 | 
			
		||||
      name: Regular
 | 
			
		||||
    - entity: light.kitchen_lights
 | 
			
		||||
      name: Regular
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoPicEntity extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-glance-card")
 | 
			
		||||
class DemoGlanceEntity extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-glance-card", DemoPicEntity);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-glance-card": DemoGlanceEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -44,6 +43,110 @@ const ENTITIES = [
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const CONFIGS = [
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Default Grid",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  cards:
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.kitchen_lights
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.bed_light
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_paulus
 | 
			
		||||
    - type: sensor
 | 
			
		||||
      entity: sensor.illumination
 | 
			
		||||
      graph: line
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_anne_therese
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Non-square Grid with 2 columns",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  columns: 2
 | 
			
		||||
  square: false
 | 
			
		||||
  cards:
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.kitchen_lights
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.bed_light
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_paulus
 | 
			
		||||
    - type: sensor
 | 
			
		||||
      entity: sensor.illumination
 | 
			
		||||
      graph: line
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Default Grid with title",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  title: Kitchen
 | 
			
		||||
  cards:
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.kitchen_lights
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.bed_light
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_paulus
 | 
			
		||||
    - type: sensor
 | 
			
		||||
      entity: sensor.illumination
 | 
			
		||||
      graph: line
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_anne_therese
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Columns 4",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  columns: 4
 | 
			
		||||
  cards:
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.kitchen_lights
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.bed_light
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_paulus
 | 
			
		||||
    - type: sensor
 | 
			
		||||
      entity: sensor.illumination
 | 
			
		||||
      graph: line
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Columns 2",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  columns: 2
 | 
			
		||||
  cards:
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.kitchen_lights
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.bed_light
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Columns 1",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  columns: 1
 | 
			
		||||
  cards:
 | 
			
		||||
  - type: entity
 | 
			
		||||
    entity: light.kitchen_lights
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Size for single card",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  cards:
 | 
			
		||||
  - type: entity
 | 
			
		||||
    entity: light.kitchen_lights
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Vertical Stack",
 | 
			
		||||
    config: `
 | 
			
		||||
@@ -94,65 +197,28 @@ const CONFIGS = [
 | 
			
		||||
      entity: light.bed_light
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Default Grid",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  cards:
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.kitchen_lights
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.bed_light
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_paulus
 | 
			
		||||
    - type: sensor
 | 
			
		||||
      entity: sensor.illumination
 | 
			
		||||
      graph: line
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_anne_therese
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Non-square Grid with 2 columns",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: grid
 | 
			
		||||
  columns: 2
 | 
			
		||||
  square: false
 | 
			
		||||
  cards:
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.kitchen_lights
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: light.bed_light
 | 
			
		||||
    - type: entity
 | 
			
		||||
      entity: device_tracker.demo_paulus
 | 
			
		||||
    - type: sensor
 | 
			
		||||
      entity: sensor.illumination
 | 
			
		||||
      graph: line
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoStack extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-grid-and-stack-card")
 | 
			
		||||
class DemoStack extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
    mockHistory(hass);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-stack-card", DemoStack);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-grid-and-stack-card": DemoStack;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
import { html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement } from "lit/decorators";
 | 
			
		||||
import "../components/demo-cards";
 | 
			
		||||
 | 
			
		||||
const CONFIGS = [
 | 
			
		||||
@@ -37,19 +36,15 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoIframe extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
@customElement("demo-hui-iframe-card")
 | 
			
		||||
class DemoIframe extends LitElement {
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-iframe-card", DemoIframe);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-iframe-card": DemoIframe;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -63,26 +62,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoLightEntity extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-light-card")
 | 
			
		||||
class DemoLightEntity extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-light-card", DemoLightEntity);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-light-card": DemoLightEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -161,33 +160,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoMap extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html`
 | 
			
		||||
      <demo-cards
 | 
			
		||||
        id="demos"
 | 
			
		||||
        hass="[[hass]]"
 | 
			
		||||
        configs="[[_configs]]"
 | 
			
		||||
      ></demo-cards>
 | 
			
		||||
    `;
 | 
			
		||||
@customElement("demo-hui-map-card")
 | 
			
		||||
class DemoMap extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
      hass: Object,
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-map-card", DemoMap);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-map-card": DemoMap;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -254,25 +253,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoMarkdown extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-markdown-card")
 | 
			
		||||
class DemoMarkdown extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    mockTemplate(hass);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-markdown-card", DemoMarkdown);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-markdown-card": DemoMarkdown;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -146,35 +145,37 @@ const CONFIGS = [
 | 
			
		||||
    entity: media_player.receiver_off
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Grid Full Size",
 | 
			
		||||
    config: `
 | 
			
		||||
  - type: grid
 | 
			
		||||
    columns: 1
 | 
			
		||||
    cards:
 | 
			
		||||
    - type: media-control
 | 
			
		||||
      entity: media_player.music_paused
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoHuiMediControlCard extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html`
 | 
			
		||||
      <demo-cards
 | 
			
		||||
        id="demos"
 | 
			
		||||
        hass="[[hass]]"
 | 
			
		||||
        configs="[[_configs]]"
 | 
			
		||||
      ></demo-cards>
 | 
			
		||||
    `;
 | 
			
		||||
@customElement("demo-hui-media-control-card")
 | 
			
		||||
class DemoHuiMediaControlCard extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
      hass: Object,
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(createMediaPlayerEntities());
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-media-control-card", DemoHuiMediControlCard);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-media-control-card": DemoHuiMediaControlCard;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -26,9 +25,9 @@ const CONFIGS = [
 | 
			
		||||
    - entity: media_player.android_cast
 | 
			
		||||
      name: Screen casting
 | 
			
		||||
    - entity: media_player.image_display
 | 
			
		||||
      name: Digital Picture Frame  
 | 
			
		||||
      name: Digital Picture Frame
 | 
			
		||||
    - entity: media_player.sonos_idle
 | 
			
		||||
      name: Sonos Idle  
 | 
			
		||||
      name: Sonos Idle
 | 
			
		||||
    - entity: media_player.idle_browse_media
 | 
			
		||||
      name: Idle waiting for Browse Media
 | 
			
		||||
    - entity: media_player.theater_off
 | 
			
		||||
@@ -38,7 +37,7 @@ const CONFIGS = [
 | 
			
		||||
    - entity: media_player.theater_off_static
 | 
			
		||||
      name: Player Off (cannot be switched on)
 | 
			
		||||
    - entity: media_player.theater_on_static
 | 
			
		||||
      name: Player On (cannot be switched off)  
 | 
			
		||||
      name: Player On (cannot be switched off)
 | 
			
		||||
    - entity: media_player.idle
 | 
			
		||||
      name: Player Idle
 | 
			
		||||
    - entity: media_player.playing
 | 
			
		||||
@@ -55,33 +54,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoHuiMediaPlayerRows extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html`
 | 
			
		||||
      <demo-cards
 | 
			
		||||
        id="demos"
 | 
			
		||||
        hass="[[hass]]"
 | 
			
		||||
        configs="[[_configs]]"
 | 
			
		||||
      ></demo-cards>
 | 
			
		||||
    `;
 | 
			
		||||
@customElement("demo-hui-media-player-row")
 | 
			
		||||
class DemoHuiMediaPlayerRow extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
      hass: Object,
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(createMediaPlayerEntities());
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-media-player-rows", DemoHuiMediaPlayerRows);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-media-player-row": DemoHuiMediaPlayerRow;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -125,26 +124,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoPicElements extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-picture-elements-card")
 | 
			
		||||
class DemoPictureElements extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-picture-elements-card", DemoPicElements);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-picture-elements-card": DemoPictureElements;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -80,26 +79,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoPicEntity extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-picture-entity-card")
 | 
			
		||||
class DemoPictureEntity extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-picture-entity-card", DemoPicEntity);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-picture-entity-card": DemoPictureEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -121,26 +120,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoPicGlance extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-picture-glance-card")
 | 
			
		||||
class DemoPictureGlance extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-picture-glance-card", DemoPicGlance);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-picture-glance-card": DemoPictureGlance;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										53
									
								
								gallery/src/demos/demo-hui-plant-card.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								gallery/src/demos/demo-hui-plant-card.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,53 @@
 | 
			
		||||
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 { createPlantEntities } from "../data/plants";
 | 
			
		||||
 | 
			
		||||
const CONFIGS = [
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Basic example",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: plant-status
 | 
			
		||||
  entity: plant.lemon_tree
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "Problem (too bright) + low battery",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: plant-status
 | 
			
		||||
  entity: plant.apple_tree
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    heading: "With picture + multiple problems",
 | 
			
		||||
    config: `
 | 
			
		||||
- type: plant-status
 | 
			
		||||
  entity: plant.sunflowers
 | 
			
		||||
  name: Sunflowers Name Overwrite
 | 
			
		||||
    `,
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
@customElement("demo-hui-plant-card")
 | 
			
		||||
export class DemoPlantEntity 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(createPlantEntities());
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-plant-card": DemoPlantEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
 | 
			
		||||
@@ -20,24 +19,19 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoShoppingListEntity extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-shopping-list-card")
 | 
			
		||||
class DemoShoppingListEntity extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
 | 
			
		||||
    hass.mockAPI("shopping_list", () => [
 | 
			
		||||
      { name: "list", id: 1, complete: false },
 | 
			
		||||
@@ -48,4 +42,8 @@ class DemoShoppingListEntity extends PolymerElement {
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-shopping-list-card", DemoShoppingListEntity);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-shopping-list-card": DemoShoppingListEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,5 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
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";
 | 
			
		||||
@@ -74,26 +73,25 @@ const CONFIGS = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoThermostatEntity extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
    return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
 | 
			
		||||
@customElement("demo-hui-thermostat-card")
 | 
			
		||||
class DemoThermostatEntity extends LitElement {
 | 
			
		||||
  @query("#demos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`<demo-cards id="demos" .configs=${CONFIGS}></demo-cards>`;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _configs: {
 | 
			
		||||
        type: Object,
 | 
			
		||||
        value: CONFIGS,
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    const hass = provideHass(this.$.demos);
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("lovelace", "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-hui-thermostat-card", DemoThermostatEntity);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-hui-thermostat-card": DemoThermostatEntity;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										354
									
								
								gallery/src/demos/demo-integration-card.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										354
									
								
								gallery/src/demos/demo-integration-card.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,354 @@
 | 
			
		||||
import { html, css, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import "../../../src/components/ha-formfield";
 | 
			
		||||
import "../../../src/components/ha-switch";
 | 
			
		||||
 | 
			
		||||
import { IntegrationManifest } from "../../../src/data/integration";
 | 
			
		||||
 | 
			
		||||
import { provideHass } from "../../../src/fake_data/provide_hass";
 | 
			
		||||
import { HomeAssistant } from "../../../src/types";
 | 
			
		||||
import "../../../src/panels/config/integrations/ha-integration-card";
 | 
			
		||||
import "../../../src/panels/config/integrations/ha-ignored-config-entry-card";
 | 
			
		||||
import "../../../src/panels/config/integrations/ha-config-flow-card";
 | 
			
		||||
import type {
 | 
			
		||||
  ConfigEntryExtended,
 | 
			
		||||
  DataEntryFlowProgressExtended,
 | 
			
		||||
} from "../../../src/panels/config/integrations/ha-config-integrations";
 | 
			
		||||
import { DeviceRegistryEntry } from "../../../src/data/device_registry";
 | 
			
		||||
import { EntityRegistryEntry } from "../../../src/data/entity_registry";
 | 
			
		||||
import { classMap } from "lit/directives/class-map";
 | 
			
		||||
import { customElement, property, state } from "lit/decorators";
 | 
			
		||||
 | 
			
		||||
const createConfigEntry = (
 | 
			
		||||
  title: string,
 | 
			
		||||
  override: Partial<ConfigEntryExtended> = {}
 | 
			
		||||
): ConfigEntryExtended => ({
 | 
			
		||||
  entry_id: title,
 | 
			
		||||
  domain: "esphome",
 | 
			
		||||
  localized_domain_name: "ESPHome",
 | 
			
		||||
  title,
 | 
			
		||||
  source: "zeroconf",
 | 
			
		||||
  state: "loaded",
 | 
			
		||||
  supports_options: false,
 | 
			
		||||
  supports_unload: true,
 | 
			
		||||
  disabled_by: null,
 | 
			
		||||
  pref_disable_new_entities: false,
 | 
			
		||||
  pref_disable_polling: false,
 | 
			
		||||
  reason: null,
 | 
			
		||||
  ...override,
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const createManifest = (
 | 
			
		||||
  isCustom: boolean,
 | 
			
		||||
  isCloud: boolean,
 | 
			
		||||
  name = "ESPHome"
 | 
			
		||||
): IntegrationManifest => ({
 | 
			
		||||
  name,
 | 
			
		||||
  domain: "esphome",
 | 
			
		||||
  is_built_in: !isCustom,
 | 
			
		||||
  config_flow: false,
 | 
			
		||||
  documentation: "https://www.home-assistant.io/integrations/esphome/",
 | 
			
		||||
  iot_class: isCloud ? "cloud_polling" : "local_polling",
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
const loadedEntry = createConfigEntry("Loaded");
 | 
			
		||||
const nameAsDomainEntry = createConfigEntry("ESPHome");
 | 
			
		||||
const longNameEntry = createConfigEntry(
 | 
			
		||||
  "Entry with a super long name that is going to the next line"
 | 
			
		||||
);
 | 
			
		||||
const longNonBreakingNameEntry = createConfigEntry(
 | 
			
		||||
  "EntryWithASuperLongNameThatDoesNotBreak"
 | 
			
		||||
);
 | 
			
		||||
const configPanelEntry = createConfigEntry("Config Panel", {
 | 
			
		||||
  domain: "mqtt",
 | 
			
		||||
  localized_domain_name: "MQTT",
 | 
			
		||||
});
 | 
			
		||||
const optionsFlowEntry = createConfigEntry("Options Flow", {
 | 
			
		||||
  supports_options: true,
 | 
			
		||||
});
 | 
			
		||||
const disabledPollingEntry = createConfigEntry("Disabled Polling", {
 | 
			
		||||
  pref_disable_polling: true,
 | 
			
		||||
});
 | 
			
		||||
const setupErrorEntry = createConfigEntry("Setup Error", {
 | 
			
		||||
  state: "setup_error",
 | 
			
		||||
});
 | 
			
		||||
const migrationErrorEntry = createConfigEntry("Migration Error", {
 | 
			
		||||
  state: "migration_error",
 | 
			
		||||
});
 | 
			
		||||
const setupRetryEntry = createConfigEntry("Setup Retry", {
 | 
			
		||||
  state: "setup_retry",
 | 
			
		||||
});
 | 
			
		||||
const setupRetryReasonEntry = createConfigEntry("Setup Retry", {
 | 
			
		||||
  state: "setup_retry",
 | 
			
		||||
  reason: "connection_error",
 | 
			
		||||
});
 | 
			
		||||
const setupRetryReasonMissingKeyEntry = createConfigEntry("Setup Retry", {
 | 
			
		||||
  state: "setup_retry",
 | 
			
		||||
  reason:
 | 
			
		||||
    "HTTPSConnectionpool: Max retries exceeded with NewConnectionError('<urllib3.connection.HTTPSConnection object at 0x9eedfc10>: Failed to establish a new connection: [Errno 113] Host is unreachable')",
 | 
			
		||||
});
 | 
			
		||||
const failedUnloadEntry = createConfigEntry("Failed Unload", {
 | 
			
		||||
  state: "failed_unload",
 | 
			
		||||
});
 | 
			
		||||
const notLoadedEntry = createConfigEntry("Not Loaded", { state: "not_loaded" });
 | 
			
		||||
const disabledEntry = createConfigEntry("Disabled", {
 | 
			
		||||
  state: "not_loaded",
 | 
			
		||||
  disabled_by: "user",
 | 
			
		||||
});
 | 
			
		||||
const disabledFailedUnloadEntry = createConfigEntry(
 | 
			
		||||
  "Disabled - Failed Unload",
 | 
			
		||||
  {
 | 
			
		||||
    state: "failed_unload",
 | 
			
		||||
    disabled_by: "user",
 | 
			
		||||
  }
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
const configFlows: DataEntryFlowProgressExtended[] = [
 | 
			
		||||
  {
 | 
			
		||||
    flow_id: "adbb401329d8439ebb78ef29837826a8",
 | 
			
		||||
    handler: "roku",
 | 
			
		||||
    context: {
 | 
			
		||||
      source: "ssdp",
 | 
			
		||||
      unique_id: "YF008D862864",
 | 
			
		||||
      title_placeholders: {
 | 
			
		||||
        name: "Living room Roku",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    step_id: "discovery_confirm",
 | 
			
		||||
    localized_title: "Living room Roku",
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    flow_id: "adbb401329d8439ebb78ef29837826a8",
 | 
			
		||||
    handler: "hue",
 | 
			
		||||
    context: {
 | 
			
		||||
      source: "reauth",
 | 
			
		||||
      unique_id: "YF008D862864",
 | 
			
		||||
      title_placeholders: {
 | 
			
		||||
        name: "Living room Roku",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
    step_id: "discovery_confirm",
 | 
			
		||||
    localized_title: "Philips Hue",
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const configEntries: Array<{
 | 
			
		||||
  items: ConfigEntryExtended[];
 | 
			
		||||
  is_custom?: boolean;
 | 
			
		||||
  disabled?: boolean;
 | 
			
		||||
  highlight?: string;
 | 
			
		||||
}> = [
 | 
			
		||||
  { items: [loadedEntry] },
 | 
			
		||||
  { items: [configPanelEntry] },
 | 
			
		||||
  { items: [optionsFlowEntry] },
 | 
			
		||||
  { items: [disabledPollingEntry] },
 | 
			
		||||
  { items: [nameAsDomainEntry] },
 | 
			
		||||
  { items: [longNameEntry] },
 | 
			
		||||
  { items: [longNonBreakingNameEntry] },
 | 
			
		||||
  { items: [setupErrorEntry] },
 | 
			
		||||
  { items: [migrationErrorEntry] },
 | 
			
		||||
  { items: [setupRetryEntry] },
 | 
			
		||||
  { items: [setupRetryReasonEntry] },
 | 
			
		||||
  { items: [setupRetryReasonMissingKeyEntry] },
 | 
			
		||||
  { items: [failedUnloadEntry] },
 | 
			
		||||
  { items: [notLoadedEntry] },
 | 
			
		||||
  {
 | 
			
		||||
    items: [
 | 
			
		||||
      loadedEntry,
 | 
			
		||||
      setupErrorEntry,
 | 
			
		||||
      migrationErrorEntry,
 | 
			
		||||
      longNameEntry,
 | 
			
		||||
      longNonBreakingNameEntry,
 | 
			
		||||
      setupRetryEntry,
 | 
			
		||||
      failedUnloadEntry,
 | 
			
		||||
      notLoadedEntry,
 | 
			
		||||
      disabledEntry,
 | 
			
		||||
      nameAsDomainEntry,
 | 
			
		||||
      configPanelEntry,
 | 
			
		||||
      optionsFlowEntry,
 | 
			
		||||
    ],
 | 
			
		||||
  },
 | 
			
		||||
  { disabled: true, items: [disabledEntry] },
 | 
			
		||||
  { disabled: true, items: [disabledFailedUnloadEntry] },
 | 
			
		||||
  {
 | 
			
		||||
    disabled: true,
 | 
			
		||||
    items: [disabledEntry, disabledFailedUnloadEntry],
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    items: [loadedEntry, configPanelEntry],
 | 
			
		||||
    highlight: "Loaded",
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const createEntityRegistryEntries = (
 | 
			
		||||
  item: ConfigEntryExtended
 | 
			
		||||
): EntityRegistryEntry[] => [
 | 
			
		||||
  {
 | 
			
		||||
    config_entry_id: item.entry_id,
 | 
			
		||||
    device_id: "mock-device-id",
 | 
			
		||||
    area_id: null,
 | 
			
		||||
    disabled_by: null,
 | 
			
		||||
    entity_id: "binary_sensor.updater",
 | 
			
		||||
    name: null,
 | 
			
		||||
    icon: null,
 | 
			
		||||
    platform: "updater",
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const createDeviceRegistryEntries = (
 | 
			
		||||
  item: ConfigEntryExtended
 | 
			
		||||
): DeviceRegistryEntry[] => [
 | 
			
		||||
  {
 | 
			
		||||
    entry_type: null,
 | 
			
		||||
    config_entries: [item.entry_id],
 | 
			
		||||
    connections: [],
 | 
			
		||||
    manufacturer: "ESPHome",
 | 
			
		||||
    model: "Mock Device",
 | 
			
		||||
    name: "Tag Reader",
 | 
			
		||||
    sw_version: null,
 | 
			
		||||
    id: "mock-device-id",
 | 
			
		||||
    identifiers: [],
 | 
			
		||||
    via_device_id: null,
 | 
			
		||||
    area_id: null,
 | 
			
		||||
    name_by_user: null,
 | 
			
		||||
    disabled_by: null,
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
@customElement("demo-integration-card")
 | 
			
		||||
export class DemoIntegrationCard extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) hass?: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @state() isCustomIntegration = false;
 | 
			
		||||
 | 
			
		||||
  @state() isCloud = false;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    if (!this.hass) {
 | 
			
		||||
      return html``;
 | 
			
		||||
    }
 | 
			
		||||
    return html`
 | 
			
		||||
      <div class="container">
 | 
			
		||||
        <div class="filters">
 | 
			
		||||
          <ha-formfield label="Custom Integration">
 | 
			
		||||
            <ha-switch @change=${this._toggleCustomIntegration}></ha-switch>
 | 
			
		||||
          </ha-formfield>
 | 
			
		||||
          <ha-formfield label="Relies on cloud">
 | 
			
		||||
            <ha-switch @change=${this._toggleCloud}></ha-switch>
 | 
			
		||||
          </ha-formfield>
 | 
			
		||||
        </div>
 | 
			
		||||
 | 
			
		||||
        <ha-ignored-config-entry-card
 | 
			
		||||
          .hass=${this.hass}
 | 
			
		||||
          .entry=${createConfigEntry("Ignored Entry")}
 | 
			
		||||
          .manifest=${createManifest(this.isCustomIntegration, this.isCloud)}
 | 
			
		||||
        ></ha-ignored-config-entry-card>
 | 
			
		||||
 | 
			
		||||
        ${configFlows.map(
 | 
			
		||||
          (flow) => html`
 | 
			
		||||
            <ha-config-flow-card
 | 
			
		||||
              .hass=${this.hass}
 | 
			
		||||
              .flow=${flow}
 | 
			
		||||
              .manifest=${createManifest(
 | 
			
		||||
                this.isCustomIntegration,
 | 
			
		||||
                this.isCloud,
 | 
			
		||||
                flow.handler === "roku" ? "Roku" : "Philips Hue"
 | 
			
		||||
              )}
 | 
			
		||||
            ></ha-config-flow-card>
 | 
			
		||||
          `
 | 
			
		||||
        )}
 | 
			
		||||
        ${configEntries.map(
 | 
			
		||||
          (info) => html`
 | 
			
		||||
            <ha-integration-card
 | 
			
		||||
              class=${classMap({
 | 
			
		||||
                highlight: info.highlight !== undefined,
 | 
			
		||||
              })}
 | 
			
		||||
              .hass=${this.hass}
 | 
			
		||||
              domain="esphome"
 | 
			
		||||
              .items=${info.items}
 | 
			
		||||
              .manifest=${createManifest(
 | 
			
		||||
                this.isCustomIntegration,
 | 
			
		||||
                this.isCloud
 | 
			
		||||
              )}
 | 
			
		||||
              .entityRegistryEntries=${createEntityRegistryEntries(
 | 
			
		||||
                info.items[0]
 | 
			
		||||
              )}
 | 
			
		||||
              .deviceRegistryEntries=${createDeviceRegistryEntries(
 | 
			
		||||
                info.items[0]
 | 
			
		||||
              )}
 | 
			
		||||
              ?disabled=${info.disabled}
 | 
			
		||||
              .selectedConfigEntryId=${info.highlight}
 | 
			
		||||
            ></ha-integration-card>
 | 
			
		||||
          `
 | 
			
		||||
        )}
 | 
			
		||||
      </div>
 | 
			
		||||
      <div class="container">
 | 
			
		||||
        <!-- One that is standalone to see how it increases height if height
 | 
			
		||||
           not defined by other cards. -->
 | 
			
		||||
        <ha-integration-card
 | 
			
		||||
          .hass=${this.hass}
 | 
			
		||||
          domain="esphome"
 | 
			
		||||
          .items=${[
 | 
			
		||||
            loadedEntry,
 | 
			
		||||
            setupErrorEntry,
 | 
			
		||||
            migrationErrorEntry,
 | 
			
		||||
            setupRetryEntry,
 | 
			
		||||
            failedUnloadEntry,
 | 
			
		||||
          ]}
 | 
			
		||||
          .manifest=${createManifest(this.isCustomIntegration, this.isCloud)}
 | 
			
		||||
          .entityRegistryEntries=${createEntityRegistryEntries(loadedEntry)}
 | 
			
		||||
          .deviceRegistryEntries=${createDeviceRegistryEntries(loadedEntry)}
 | 
			
		||||
        ></ha-integration-card>
 | 
			
		||||
      </div>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  protected firstUpdated(changedProps) {
 | 
			
		||||
    super.firstUpdated(changedProps);
 | 
			
		||||
    const hass = provideHass(this);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.updateTranslations("config", "en");
 | 
			
		||||
    // Normally this string is loaded from backend
 | 
			
		||||
    hass.addTranslations(
 | 
			
		||||
      {
 | 
			
		||||
        "component.esphome.config.error.connection_error":
 | 
			
		||||
          "Can't connect to ESP. Please make sure your YAML file contains an 'api:' line.",
 | 
			
		||||
      },
 | 
			
		||||
      "en"
 | 
			
		||||
    );
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private _toggleCustomIntegration() {
 | 
			
		||||
    this.isCustomIntegration = !this.isCustomIntegration;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private _toggleCloud() {
 | 
			
		||||
    this.isCloud = !this.isCloud;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles() {
 | 
			
		||||
    return css`
 | 
			
		||||
      .container {
 | 
			
		||||
        display: grid;
 | 
			
		||||
        grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
 | 
			
		||||
        grid-gap: 16px 16px;
 | 
			
		||||
        padding: 8px 16px 16px;
 | 
			
		||||
        margin-bottom: 64px;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      .container > * {
 | 
			
		||||
        max-width: 500px;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      ha-formfield {
 | 
			
		||||
        margin: 8px 0;
 | 
			
		||||
        display: block;
 | 
			
		||||
      }
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-integration-card": DemoIntegrationCard;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,12 +1,19 @@
 | 
			
		||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
 | 
			
		||||
/* eslint-plugin-disable lit */
 | 
			
		||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
 | 
			
		||||
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, property, query } from "lit/decorators";
 | 
			
		||||
import "../../../src/components/ha-card";
 | 
			
		||||
import { SUPPORT_BRIGHTNESS } from "../../../src/data/light";
 | 
			
		||||
import { getEntity } from "../../../src/fake_data/entity";
 | 
			
		||||
import { provideHass } from "../../../src/fake_data/provide_hass";
 | 
			
		||||
import "../components/demo-more-infos";
 | 
			
		||||
import {
 | 
			
		||||
  LightColorModes,
 | 
			
		||||
  SUPPORT_EFFECT,
 | 
			
		||||
  SUPPORT_FLASH,
 | 
			
		||||
  SUPPORT_TRANSITION,
 | 
			
		||||
} from "../../../src/data/light";
 | 
			
		||||
import "../../../src/dialogs/more-info/more-info-content";
 | 
			
		||||
import { getEntity } from "../../../src/fake_data/entity";
 | 
			
		||||
import {
 | 
			
		||||
  MockHomeAssistant,
 | 
			
		||||
  provideHass,
 | 
			
		||||
} from "../../../src/fake_data/provide_hass";
 | 
			
		||||
import "../components/demo-more-infos";
 | 
			
		||||
 | 
			
		||||
const ENTITIES = [
 | 
			
		||||
  getEntity("light", "bed_light", "on", {
 | 
			
		||||
@@ -14,40 +21,135 @@ const ENTITIES = [
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("light", "kitchen_light", "on", {
 | 
			
		||||
    friendly_name: "Brightness Light",
 | 
			
		||||
    brightness: 80,
 | 
			
		||||
    supported_features: SUPPORT_BRIGHTNESS,
 | 
			
		||||
    brightness: 200,
 | 
			
		||||
    supported_color_modes: [LightColorModes.BRIGHTNESS],
 | 
			
		||||
    color_mode: LightColorModes.BRIGHTNESS,
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("light", "color_temperature_light", "on", {
 | 
			
		||||
    friendly_name: "White Color Temperature Light",
 | 
			
		||||
    brightness: 128,
 | 
			
		||||
    color_temp: 75,
 | 
			
		||||
    min_mireds: 30,
 | 
			
		||||
    max_mireds: 150,
 | 
			
		||||
    supported_color_modes: [
 | 
			
		||||
      LightColorModes.BRIGHTNESS,
 | 
			
		||||
      LightColorModes.COLOR_TEMP,
 | 
			
		||||
    ],
 | 
			
		||||
    color_mode: LightColorModes.COLOR_TEMP,
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("light", "color_hs_light", "on", {
 | 
			
		||||
    friendly_name: "Color HS Light",
 | 
			
		||||
    brightness: 255,
 | 
			
		||||
    hs_color: [30, 100],
 | 
			
		||||
    rgb_color: [30, 100, 255],
 | 
			
		||||
    min_mireds: 30,
 | 
			
		||||
    max_mireds: 150,
 | 
			
		||||
    supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
 | 
			
		||||
    supported_color_modes: [
 | 
			
		||||
      LightColorModes.BRIGHTNESS,
 | 
			
		||||
      LightColorModes.COLOR_TEMP,
 | 
			
		||||
      LightColorModes.HS,
 | 
			
		||||
    ],
 | 
			
		||||
    color_mode: LightColorModes.HS,
 | 
			
		||||
    effect_list: ["random", "colorloop"],
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("light", "color_rgb_ct_light", "on", {
 | 
			
		||||
    friendly_name: "Color RGB + CT Light",
 | 
			
		||||
    brightness: 255,
 | 
			
		||||
    color_temp: 75,
 | 
			
		||||
    min_mireds: 30,
 | 
			
		||||
    max_mireds: 150,
 | 
			
		||||
    supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
 | 
			
		||||
    supported_color_modes: [
 | 
			
		||||
      LightColorModes.BRIGHTNESS,
 | 
			
		||||
      LightColorModes.COLOR_TEMP,
 | 
			
		||||
      LightColorModes.RGB,
 | 
			
		||||
    ],
 | 
			
		||||
    color_mode: LightColorModes.COLOR_TEMP,
 | 
			
		||||
    effect_list: ["random", "colorloop"],
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("light", "color_RGB_light", "on", {
 | 
			
		||||
    friendly_name: "Color Effets Light",
 | 
			
		||||
    brightness: 255,
 | 
			
		||||
    rgb_color: [30, 100, 255],
 | 
			
		||||
    supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
 | 
			
		||||
    supported_color_modes: [LightColorModes.BRIGHTNESS, LightColorModes.RGB],
 | 
			
		||||
    color_mode: LightColorModes.RGB,
 | 
			
		||||
    effect_list: ["random", "colorloop"],
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("light", "color_rgbw_light", "on", {
 | 
			
		||||
    friendly_name: "Color RGBW Light",
 | 
			
		||||
    brightness: 255,
 | 
			
		||||
    rgbw_color: [30, 100, 255, 125],
 | 
			
		||||
    min_mireds: 30,
 | 
			
		||||
    max_mireds: 150,
 | 
			
		||||
    supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
 | 
			
		||||
    supported_color_modes: [
 | 
			
		||||
      LightColorModes.BRIGHTNESS,
 | 
			
		||||
      LightColorModes.COLOR_TEMP,
 | 
			
		||||
      LightColorModes.RGBW,
 | 
			
		||||
    ],
 | 
			
		||||
    color_mode: LightColorModes.RGBW,
 | 
			
		||||
    effect_list: ["random", "colorloop"],
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("light", "color_rgbww_light", "on", {
 | 
			
		||||
    friendly_name: "Color RGBWW Light",
 | 
			
		||||
    brightness: 255,
 | 
			
		||||
    rgbww_color: [30, 100, 255, 125, 10],
 | 
			
		||||
    min_mireds: 30,
 | 
			
		||||
    max_mireds: 150,
 | 
			
		||||
    supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
 | 
			
		||||
    supported_color_modes: [
 | 
			
		||||
      LightColorModes.BRIGHTNESS,
 | 
			
		||||
      LightColorModes.COLOR_TEMP,
 | 
			
		||||
      LightColorModes.RGBWW,
 | 
			
		||||
    ],
 | 
			
		||||
    color_mode: LightColorModes.RGBWW,
 | 
			
		||||
    effect_list: ["random", "colorloop"],
 | 
			
		||||
  }),
 | 
			
		||||
  getEntity("light", "color_xy_light", "on", {
 | 
			
		||||
    friendly_name: "Color XY Light",
 | 
			
		||||
    brightness: 255,
 | 
			
		||||
    xy_color: [30, 100],
 | 
			
		||||
    rgb_color: [30, 100, 255],
 | 
			
		||||
    min_mireds: 30,
 | 
			
		||||
    max_mireds: 150,
 | 
			
		||||
    supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
 | 
			
		||||
    supported_color_modes: [
 | 
			
		||||
      LightColorModes.BRIGHTNESS,
 | 
			
		||||
      LightColorModes.COLOR_TEMP,
 | 
			
		||||
      LightColorModes.XY,
 | 
			
		||||
    ],
 | 
			
		||||
    color_mode: LightColorModes.XY,
 | 
			
		||||
    effect_list: ["random", "colorloop"],
 | 
			
		||||
  }),
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
class DemoMoreInfoLight extends PolymerElement {
 | 
			
		||||
  static get template() {
 | 
			
		||||
@customElement("demo-more-info-light")
 | 
			
		||||
class DemoMoreInfoLight extends LitElement {
 | 
			
		||||
  @property() public hass!: MockHomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @query("demo-more-infos") private _demoRoot!: HTMLElement;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`
 | 
			
		||||
      <demo-more-infos
 | 
			
		||||
        hass="[[hass]]"
 | 
			
		||||
        entities="[[_entities]]"
 | 
			
		||||
        .hass=${this.hass}
 | 
			
		||||
        .entities=${ENTITIES.map((ent) => ent.entityId)}
 | 
			
		||||
      ></demo-more-infos>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get properties() {
 | 
			
		||||
    return {
 | 
			
		||||
      _entities: {
 | 
			
		||||
        type: Array,
 | 
			
		||||
        value: ENTITIES.map((ent) => ent.entityId),
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  public ready() {
 | 
			
		||||
    super.ready();
 | 
			
		||||
    this._setupDemo();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _setupDemo() {
 | 
			
		||||
    const hass = provideHass(this);
 | 
			
		||||
    await hass.updateTranslations(null, "en");
 | 
			
		||||
  protected firstUpdated(changedProperties: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProperties);
 | 
			
		||||
    const hass = provideHass(this._demoRoot);
 | 
			
		||||
    hass.updateTranslations(null, "en");
 | 
			
		||||
    hass.addEntities(ENTITIES);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-more-info-light", DemoMoreInfoLight);
 | 
			
		||||
declare global {
 | 
			
		||||
  interface HTMLElementTagNameMap {
 | 
			
		||||
    "demo-more-info-light": DemoMoreInfoLight;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,11 @@
 | 
			
		||||
import "@material/mwc-button";
 | 
			
		||||
import { html, LitElement, TemplateResult } from "lit-element";
 | 
			
		||||
import { html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement } from "lit/decorators";
 | 
			
		||||
import "../../../src/components/ha-card";
 | 
			
		||||
import { ActionHandlerEvent } from "../../../src/data/lovelace";
 | 
			
		||||
import { actionHandler } from "../../../src/panels/lovelace/common/directives/action-handler-directive";
 | 
			
		||||
 | 
			
		||||
@customElement("demo-util-long-press")
 | 
			
		||||
export class DemoUtilLongPress extends LitElement {
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`
 | 
			
		||||
@@ -20,7 +22,7 @@ export class DemoUtilLongPress extends LitElement {
 | 
			
		||||
 | 
			
		||||
            <textarea></textarea>
 | 
			
		||||
 | 
			
		||||
            <div>(try pressing and scrolling too!)</div>
 | 
			
		||||
            <div>Try pressing and scrolling too!</div>
 | 
			
		||||
          </ha-card>
 | 
			
		||||
        `
 | 
			
		||||
      )}
 | 
			
		||||
@@ -62,5 +64,3 @@ export class DemoUtilLongPress extends LitElement {
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
customElements.define("demo-util-long-press", DemoUtilLongPress);
 | 
			
		||||
 
 | 
			
		||||
@@ -14,54 +14,51 @@ import "../../src/styles/polymer-ha-style";
 | 
			
		||||
// eslint-disable-next-line import/extensions
 | 
			
		||||
import { DEMOS } from "../build/import-demos";
 | 
			
		||||
 | 
			
		||||
const fixPath = (path) => path.substr(2, path.length - 5);
 | 
			
		||||
 | 
			
		||||
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;
 | 
			
		||||
      }
 | 
			
		||||
        :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 {
 | 
			
		||||
          display: flex;
 | 
			
		||||
          flex-wrap: wrap;
 | 
			
		||||
          justify-content: center;
 | 
			
		||||
          align-items: start;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
      .pickers ha-card {
 | 
			
		||||
        width: 400px;
 | 
			
		||||
        display: block;
 | 
			
		||||
        margin: 16px 8px;
 | 
			
		||||
      }
 | 
			
		||||
        .pickers ha-card {
 | 
			
		||||
          width: 400px;
 | 
			
		||||
          display: block;
 | 
			
		||||
          margin: 16px 8px;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
      .pickers ha-card:last-child {
 | 
			
		||||
        margin-bottom: 16px;
 | 
			
		||||
      }
 | 
			
		||||
        .pickers ha-card:last-child {
 | 
			
		||||
          margin-bottom: 16px;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
      .intro {
 | 
			
		||||
        margin: -1em 0;
 | 
			
		||||
      }
 | 
			
		||||
        .intro {
 | 
			
		||||
          margin: -1em 0;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
      p a {
 | 
			
		||||
        color: var(--primary-color);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      a {
 | 
			
		||||
        color: var(--primary-text-color);
 | 
			
		||||
        text-decoration: none;
 | 
			
		||||
      }
 | 
			
		||||
        p a {
 | 
			
		||||
          color: var(--primary-color);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        a {
 | 
			
		||||
          color: var(--primary-text-color);
 | 
			
		||||
          text-decoration: none;
 | 
			
		||||
        }
 | 
			
		||||
      </style>
 | 
			
		||||
 | 
			
		||||
      <app-header-layout>
 | 
			
		||||
@@ -70,32 +67,42 @@ class HaGallery extends PolymerElement {
 | 
			
		||||
            <ha-icon-button
 | 
			
		||||
              icon="hass:arrow-left"
 | 
			
		||||
              on-click="_backTapped"
 | 
			
		||||
              class$='[[_computeHeaderButtonClass(_demo)]]'
 | 
			
		||||
              class$="[[_computeHeaderButtonClass(_demo)]]"
 | 
			
		||||
            ></ha-icon-button>
 | 
			
		||||
            <div main-title>[[_withDefault(_demo, "Home Assistant Gallery")]]</div>
 | 
			
		||||
            <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'>
 | 
			
		||||
        <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.
 | 
			
		||||
                    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.
 | 
			
		||||
                    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.</a>.
 | 
			
		||||
                    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]]'>
 | 
			
		||||
                <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>
 | 
			
		||||
@@ -104,30 +111,10 @@ class HaGallery extends PolymerElement {
 | 
			
		||||
                </template>
 | 
			
		||||
              </ha-card>
 | 
			
		||||
 | 
			
		||||
              <ha-card header="More Info demos">
 | 
			
		||||
                <div class='card-content intro'>
 | 
			
		||||
                  <p>
 | 
			
		||||
                    More info screens show up when an entity is clicked.
 | 
			
		||||
                  </p>
 | 
			
		||||
                </div>
 | 
			
		||||
                <template is='dom-repeat' items='[[_moreInfoDemos]]'>
 | 
			
		||||
                  <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="Util demos">
 | 
			
		||||
                <div class='card-content intro'>
 | 
			
		||||
                  <p>
 | 
			
		||||
                    Test pages for our utility functions.
 | 
			
		||||
                  </p>
 | 
			
		||||
                </div>
 | 
			
		||||
                <template is='dom-repeat' items='[[_utilDemos]]'>
 | 
			
		||||
                  <a href='#[[item]]'>
 | 
			
		||||
              <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>
 | 
			
		||||
@@ -139,7 +126,10 @@ class HaGallery extends PolymerElement {
 | 
			
		||||
          </template>
 | 
			
		||||
        </div>
 | 
			
		||||
      </app-header-layout>
 | 
			
		||||
      <notification-manager hass=[[_fakeHass]] id='notifications'></notification-manager>
 | 
			
		||||
      <notification-manager
 | 
			
		||||
        hass="[[_fakeHass]]"
 | 
			
		||||
        id="notifications"
 | 
			
		||||
      ></notification-manager>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -168,13 +158,9 @@ class HaGallery extends PolymerElement {
 | 
			
		||||
        type: Array,
 | 
			
		||||
        computed: "_computeLovelace(_demos)",
 | 
			
		||||
      },
 | 
			
		||||
      _moreInfoDemos: {
 | 
			
		||||
      _restDemos: {
 | 
			
		||||
        type: Array,
 | 
			
		||||
        computed: "_computeMoreInfos(_demos)",
 | 
			
		||||
      },
 | 
			
		||||
      _utilDemos: {
 | 
			
		||||
        type: Array,
 | 
			
		||||
        computed: "_computeUtil(_demos)",
 | 
			
		||||
        computed: "_computeRest(_demos)",
 | 
			
		||||
      },
 | 
			
		||||
    };
 | 
			
		||||
  }
 | 
			
		||||
@@ -227,12 +213,8 @@ class HaGallery extends PolymerElement {
 | 
			
		||||
    return demos.filter((demo) => demo.includes("hui"));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  _computeMoreInfos(demos) {
 | 
			
		||||
    return demos.filter((demo) => demo.includes("more-info"));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  _computeUtil(demos) {
 | 
			
		||||
    return demos.filter((demo) => demo.includes("util"));
 | 
			
		||||
  _computeRest(demos) {
 | 
			
		||||
    return demos.filter((demo) => !demo.includes("hui"));
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,12 +1,6 @@
 | 
			
		||||
import { mdiArrowUpBoldCircle, mdiPuzzle } from "@mdi/js";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResultArray,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { property } from "lit/decorators";
 | 
			
		||||
import memoizeOne from "memoize-one";
 | 
			
		||||
import { atLeastVersion } from "../../../src/common/config/version";
 | 
			
		||||
import { navigate } from "../../../src/common/navigate";
 | 
			
		||||
@@ -15,6 +9,7 @@ import {
 | 
			
		||||
  HassioAddonInfo,
 | 
			
		||||
  HassioAddonRepository,
 | 
			
		||||
} from "../../../src/data/hassio/addon";
 | 
			
		||||
import { Supervisor } from "../../../src/data/supervisor/supervisor";
 | 
			
		||||
import { HomeAssistant } from "../../../src/types";
 | 
			
		||||
import "../components/hassio-card-content";
 | 
			
		||||
import { filterAndSort } from "../components/hassio-filter-addons";
 | 
			
		||||
@@ -23,6 +18,8 @@ import { hassioStyle } from "../resources/hassio-style";
 | 
			
		||||
class HassioAddonRepositoryEl extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) public hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public supervisor!: Supervisor;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public repo!: HassioAddonRepository;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public addons!: HassioAddonInfo[];
 | 
			
		||||
@@ -44,9 +41,7 @@ class HassioAddonRepositoryEl extends LitElement {
 | 
			
		||||
    const repo = this.repo;
 | 
			
		||||
    let _addons = this.addons;
 | 
			
		||||
    if (!this.hass.userData?.showAdvanced) {
 | 
			
		||||
      _addons = _addons.filter((addon) => {
 | 
			
		||||
        return !addon.advanced;
 | 
			
		||||
      });
 | 
			
		||||
      _addons = _addons.filter((addon) => !addon.advanced);
 | 
			
		||||
    }
 | 
			
		||||
    const addons = this._getAddons(_addons, this.filter);
 | 
			
		||||
 | 
			
		||||
@@ -54,16 +49,18 @@ class HassioAddonRepositoryEl extends LitElement {
 | 
			
		||||
      return html`
 | 
			
		||||
        <div class="content">
 | 
			
		||||
          <p class="description">
 | 
			
		||||
            No results found in "${repo.name}."
 | 
			
		||||
            ${this.supervisor.localize(
 | 
			
		||||
              "store.no_results_found",
 | 
			
		||||
              "repository",
 | 
			
		||||
              repo.name
 | 
			
		||||
            )}
 | 
			
		||||
          </p>
 | 
			
		||||
        </div>
 | 
			
		||||
      `;
 | 
			
		||||
    }
 | 
			
		||||
    return html`
 | 
			
		||||
      <div class="content">
 | 
			
		||||
        <h1>
 | 
			
		||||
          ${repo.name}
 | 
			
		||||
        </h1>
 | 
			
		||||
        <h1>${repo.name}</h1>
 | 
			
		||||
        <div class="card-group">
 | 
			
		||||
          ${addons.map(
 | 
			
		||||
            (addon) => html`
 | 
			
		||||
@@ -83,11 +80,13 @@ class HassioAddonRepositoryEl extends LitElement {
 | 
			
		||||
                      : mdiPuzzle}
 | 
			
		||||
                    .iconTitle=${addon.installed
 | 
			
		||||
                      ? addon.update_available
 | 
			
		||||
                        ? "New version available"
 | 
			
		||||
                        : "Add-on is installed"
 | 
			
		||||
                        ? this.supervisor.localize(
 | 
			
		||||
                            "common.new_version_available"
 | 
			
		||||
                          )
 | 
			
		||||
                        : this.supervisor.localize("addon.installed")
 | 
			
		||||
                      : addon.available
 | 
			
		||||
                      ? "Add-on is not installed"
 | 
			
		||||
                      : "Add-on is not available on your system"}
 | 
			
		||||
                      ? this.supervisor.localize("addon.not_installed")
 | 
			
		||||
                      : this.supervisor.localize("addon.not_available")}
 | 
			
		||||
                    .iconClass=${addon.installed
 | 
			
		||||
                      ? addon.update_available
 | 
			
		||||
                        ? "update"
 | 
			
		||||
@@ -121,10 +120,10 @@ class HassioAddonRepositoryEl extends LitElement {
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private _addonTapped(ev) {
 | 
			
		||||
    navigate(this, `/hassio/addon/${ev.currentTarget.addon.slug}`);
 | 
			
		||||
    navigate(`/hassio/addon/${ev.currentTarget.addon.slug}`);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResultArray {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return [
 | 
			
		||||
      hassioStyle,
 | 
			
		||||
      css`
 | 
			
		||||
 
 | 
			
		||||
@@ -4,24 +4,27 @@ import "@material/mwc-list/mwc-list-item";
 | 
			
		||||
import { mdiDotsVertical } from "@mdi/js";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  CSSResultGroup,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  PropertyValues,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { html, TemplateResult } from "lit-html";
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit";
 | 
			
		||||
import { property, state } from "lit/decorators";
 | 
			
		||||
import memoizeOne from "memoize-one";
 | 
			
		||||
import { atLeastVersion } from "../../../src/common/config/version";
 | 
			
		||||
import { fireEvent } from "../../../src/common/dom/fire_event";
 | 
			
		||||
import { navigate } from "../../../src/common/navigate";
 | 
			
		||||
import "../../../src/common/search/search-input";
 | 
			
		||||
import { extractSearchParam } from "../../../src/common/url/search-params";
 | 
			
		||||
import "../../../src/components/ha-button-menu";
 | 
			
		||||
import "../../../src/components/ha-svg-icon";
 | 
			
		||||
import {
 | 
			
		||||
  fetchHassioAddonsInfo,
 | 
			
		||||
  HassioAddonInfo,
 | 
			
		||||
  HassioAddonRepository,
 | 
			
		||||
  reloadHassioAddons,
 | 
			
		||||
} from "../../../src/data/hassio/addon";
 | 
			
		||||
import { extractApiErrorMessage } from "../../../src/data/hassio/common";
 | 
			
		||||
import { Supervisor } from "../../../src/data/supervisor/supervisor";
 | 
			
		||||
import "../../../src/layouts/hass-loading-screen";
 | 
			
		||||
import "../../../src/layouts/hass-tabs-subpage";
 | 
			
		||||
import { HomeAssistant, Route } from "../../../src/types";
 | 
			
		||||
@@ -49,58 +52,41 @@ const sortRepos = (a: HassioAddonRepository, b: HassioAddonRepository) => {
 | 
			
		||||
class HassioAddonStore extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) public hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public supervisor!: Supervisor;
 | 
			
		||||
 | 
			
		||||
  @property({ type: Boolean }) public narrow!: boolean;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public route!: Route;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) private _addons?: HassioAddonInfo[];
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) private _repos?: HassioAddonRepository[];
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _filter?: string;
 | 
			
		||||
  @state() private _filter?: string;
 | 
			
		||||
 | 
			
		||||
  public async refreshData() {
 | 
			
		||||
    this._repos = undefined;
 | 
			
		||||
    this._addons = undefined;
 | 
			
		||||
    this._filter = undefined;
 | 
			
		||||
    await reloadHassioAddons(this.hass);
 | 
			
		||||
    await this._loadData();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    const repos: TemplateResult[] = [];
 | 
			
		||||
    let repos: TemplateResult[] = [];
 | 
			
		||||
 | 
			
		||||
    if (this._repos) {
 | 
			
		||||
      for (const repo of this._repos) {
 | 
			
		||||
        const addons = this._addons!.filter(
 | 
			
		||||
          (addon) => addon.repository === repo.slug
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        if (addons.length === 0) {
 | 
			
		||||
          continue;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        repos.push(html`
 | 
			
		||||
          <hassio-addon-repository
 | 
			
		||||
            .hass=${this.hass}
 | 
			
		||||
            .repo=${repo}
 | 
			
		||||
            .addons=${addons}
 | 
			
		||||
            .filter=${this._filter!}
 | 
			
		||||
          ></hassio-addon-repository>
 | 
			
		||||
        `);
 | 
			
		||||
      }
 | 
			
		||||
    if (this.supervisor.addon.repositories) {
 | 
			
		||||
      repos = this.addonRepositories(
 | 
			
		||||
        this.supervisor.addon.repositories,
 | 
			
		||||
        this.supervisor.addon.addons,
 | 
			
		||||
        this._filter
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return html`
 | 
			
		||||
      <hass-tabs-subpage
 | 
			
		||||
        .hass=${this.hass}
 | 
			
		||||
        .localizeFunc=${this.supervisor.localize}
 | 
			
		||||
        .narrow=${this.narrow}
 | 
			
		||||
        .route=${this.route}
 | 
			
		||||
        hassio
 | 
			
		||||
        main-page
 | 
			
		||||
        .tabs=${supervisorTabs}
 | 
			
		||||
        main-page
 | 
			
		||||
        supervisor
 | 
			
		||||
      >
 | 
			
		||||
        <span slot="header">Add-on Store</span>
 | 
			
		||||
        <span slot="header"> ${this.supervisor.localize("panel.store")} </span>
 | 
			
		||||
        <ha-button-menu
 | 
			
		||||
          corner="BOTTOM_START"
 | 
			
		||||
          slot="toolbar-icon"
 | 
			
		||||
@@ -110,15 +96,15 @@ class HassioAddonStore extends LitElement {
 | 
			
		||||
            <ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon>
 | 
			
		||||
          </mwc-icon-button>
 | 
			
		||||
          <mwc-list-item>
 | 
			
		||||
            Repositories
 | 
			
		||||
            ${this.supervisor.localize("store.repositories")}
 | 
			
		||||
          </mwc-list-item>
 | 
			
		||||
          <mwc-list-item>
 | 
			
		||||
            Reload
 | 
			
		||||
            ${this.supervisor.localize("common.reload")}
 | 
			
		||||
          </mwc-list-item>
 | 
			
		||||
          ${this.hass.userData?.showAdvanced &&
 | 
			
		||||
          atLeastVersion(this.hass.config.version, 0, 117)
 | 
			
		||||
            ? html`<mwc-list-item>
 | 
			
		||||
                Registries
 | 
			
		||||
                ${this.supervisor.localize("store.registries")}
 | 
			
		||||
              </mwc-list-item>`
 | 
			
		||||
            : ""}
 | 
			
		||||
        </ha-button-menu>
 | 
			
		||||
@@ -139,11 +125,9 @@ class HassioAddonStore extends LitElement {
 | 
			
		||||
        ${!this.hass.userData?.showAdvanced
 | 
			
		||||
          ? html`
 | 
			
		||||
              <div class="advanced">
 | 
			
		||||
                Missing add-ons? Enable advanced mode on
 | 
			
		||||
                <a href="/profile" target="_top">
 | 
			
		||||
                  your profile page
 | 
			
		||||
                  ${this.supervisor.localize("store.missing_addons")}
 | 
			
		||||
                </a>
 | 
			
		||||
                .
 | 
			
		||||
              </div>
 | 
			
		||||
            `
 | 
			
		||||
          : ""}
 | 
			
		||||
@@ -153,14 +137,45 @@ class HassioAddonStore extends LitElement {
 | 
			
		||||
 | 
			
		||||
  protected firstUpdated(changedProps: PropertyValues) {
 | 
			
		||||
    super.firstUpdated(changedProps);
 | 
			
		||||
    const repositoryUrl = extractSearchParam("repository_url");
 | 
			
		||||
    navigate("/hassio/store", { replace: true });
 | 
			
		||||
    if (repositoryUrl) {
 | 
			
		||||
      this._manageRepositories(repositoryUrl);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    this.addEventListener("hass-api-called", (ev) => this.apiCalled(ev));
 | 
			
		||||
    this._loadData();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private addonRepositories = memoizeOne(
 | 
			
		||||
    (
 | 
			
		||||
      repositories: HassioAddonRepository[],
 | 
			
		||||
      addons: HassioAddonInfo[],
 | 
			
		||||
      filter?: string
 | 
			
		||||
    ) =>
 | 
			
		||||
      repositories.sort(sortRepos).map((repo) => {
 | 
			
		||||
        const filteredAddons = addons.filter(
 | 
			
		||||
          (addon) => addon.repository === repo.slug
 | 
			
		||||
        );
 | 
			
		||||
 | 
			
		||||
        return filteredAddons.length !== 0
 | 
			
		||||
          ? html`
 | 
			
		||||
              <hassio-addon-repository
 | 
			
		||||
                .hass=${this.hass}
 | 
			
		||||
                .repo=${repo}
 | 
			
		||||
                .addons=${filteredAddons}
 | 
			
		||||
                .filter=${filter!}
 | 
			
		||||
                .supervisor=${this.supervisor}
 | 
			
		||||
              ></hassio-addon-repository>
 | 
			
		||||
            `
 | 
			
		||||
          : html``;
 | 
			
		||||
      })
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  private _handleAction(ev: CustomEvent<ActionDetail>) {
 | 
			
		||||
    switch (ev.detail.index) {
 | 
			
		||||
      case 0:
 | 
			
		||||
        this._manageRepositories();
 | 
			
		||||
        this._manageRepositoriesClicked();
 | 
			
		||||
        break;
 | 
			
		||||
      case 1:
 | 
			
		||||
        this.refreshData();
 | 
			
		||||
@@ -177,33 +192,33 @@ class HassioAddonStore extends LitElement {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _manageRepositories() {
 | 
			
		||||
  private _manageRepositoriesClicked() {
 | 
			
		||||
    this._manageRepositories();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _manageRepositories(url?: string) {
 | 
			
		||||
    showRepositoriesDialog(this, {
 | 
			
		||||
      repos: this._repos!,
 | 
			
		||||
      loadData: () => this._loadData(),
 | 
			
		||||
      supervisor: this.supervisor,
 | 
			
		||||
      url,
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _manageRegistries() {
 | 
			
		||||
    showRegistriesDialog(this);
 | 
			
		||||
    showRegistriesDialog(this, { supervisor: this.supervisor });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _loadData() {
 | 
			
		||||
    try {
 | 
			
		||||
      const addonsInfo = await fetchHassioAddonsInfo(this.hass);
 | 
			
		||||
      this._repos = addonsInfo.repositories;
 | 
			
		||||
      this._repos.sort(sortRepos);
 | 
			
		||||
      this._addons = addonsInfo.addons;
 | 
			
		||||
    } catch (err) {
 | 
			
		||||
      alert(extractApiErrorMessage(err));
 | 
			
		||||
    }
 | 
			
		||||
    fireEvent(this, "supervisor-collection-refresh", { collection: "addon" });
 | 
			
		||||
    fireEvent(this, "supervisor-collection-refresh", {
 | 
			
		||||
      collection: "supervisor",
 | 
			
		||||
    });
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _filterChanged(e) {
 | 
			
		||||
    this._filter = e.detail.value;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return css`
 | 
			
		||||
      hassio-addon-repository {
 | 
			
		||||
        margin-top: 24px;
 | 
			
		||||
 
 | 
			
		||||
@@ -4,16 +4,15 @@ import "@polymer/paper-item/paper-item";
 | 
			
		||||
import "@polymer/paper-listbox/paper-listbox";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  CSSResultGroup,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  PropertyValues,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
} from "lit";
 | 
			
		||||
import { customElement, property, state } from "lit/decorators";
 | 
			
		||||
import "web-animations-js/web-animations-next-lite.min";
 | 
			
		||||
import "../../../../src/components/buttons/ha-progress-button";
 | 
			
		||||
import "../../../../src/components/ha-card";
 | 
			
		||||
import {
 | 
			
		||||
  HassioAddonDetails,
 | 
			
		||||
@@ -24,36 +23,42 @@ import {
 | 
			
		||||
  fetchHassioHardwareAudio,
 | 
			
		||||
  HassioHardwareAudioDevice,
 | 
			
		||||
} from "../../../../src/data/hassio/hardware";
 | 
			
		||||
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
 | 
			
		||||
import { haStyle } from "../../../../src/resources/styles";
 | 
			
		||||
import { HomeAssistant } from "../../../../src/types";
 | 
			
		||||
import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart";
 | 
			
		||||
import { hassioStyle } from "../../resources/hassio-style";
 | 
			
		||||
import "../../../../src/components/buttons/ha-progress-button";
 | 
			
		||||
 | 
			
		||||
@customElement("hassio-addon-audio")
 | 
			
		||||
class HassioAddonAudio extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) public hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public supervisor!: Supervisor;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public addon!: HassioAddonDetails;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _error?: string;
 | 
			
		||||
  @state() private _error?: string;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _inputDevices?: HassioHardwareAudioDevice[];
 | 
			
		||||
  @state() private _inputDevices?: HassioHardwareAudioDevice[];
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _outputDevices?: HassioHardwareAudioDevice[];
 | 
			
		||||
  @state() private _outputDevices?: HassioHardwareAudioDevice[];
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _selectedInput!: null | string;
 | 
			
		||||
  @state() private _selectedInput!: null | string;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _selectedOutput!: null | string;
 | 
			
		||||
  @state() private _selectedOutput!: null | string;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    return html`
 | 
			
		||||
      <ha-card header="Audio">
 | 
			
		||||
      <ha-card
 | 
			
		||||
        .header=${this.supervisor.localize("addon.configuration.audio.header")}
 | 
			
		||||
      >
 | 
			
		||||
        <div class="card-content">
 | 
			
		||||
          ${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
 | 
			
		||||
 | 
			
		||||
          <paper-dropdown-menu
 | 
			
		||||
            label="Input"
 | 
			
		||||
            .label=${this.supervisor.localize(
 | 
			
		||||
              "addon.configuration.audio.input"
 | 
			
		||||
            )}
 | 
			
		||||
            @iron-select=${this._setInputDevice}
 | 
			
		||||
          >
 | 
			
		||||
            <paper-listbox
 | 
			
		||||
@@ -62,17 +67,19 @@ class HassioAddonAudio extends LitElement {
 | 
			
		||||
              .selected=${this._selectedInput!}
 | 
			
		||||
            >
 | 
			
		||||
              ${this._inputDevices &&
 | 
			
		||||
              this._inputDevices.map((item) => {
 | 
			
		||||
                return html`
 | 
			
		||||
                  <paper-item device=${item.device || ""}
 | 
			
		||||
                    >${item.name}</paper-item
 | 
			
		||||
                  >
 | 
			
		||||
                `;
 | 
			
		||||
              })}
 | 
			
		||||
              this._inputDevices.map(
 | 
			
		||||
                (item) => html`
 | 
			
		||||
                  <paper-item device=${item.device || ""}>
 | 
			
		||||
                    ${item.name}
 | 
			
		||||
                  </paper-item>
 | 
			
		||||
                `
 | 
			
		||||
              )}
 | 
			
		||||
            </paper-listbox>
 | 
			
		||||
          </paper-dropdown-menu>
 | 
			
		||||
          <paper-dropdown-menu
 | 
			
		||||
            label="Output"
 | 
			
		||||
            .label=${this.supervisor.localize(
 | 
			
		||||
              "addon.configuration.audio.output"
 | 
			
		||||
            )}
 | 
			
		||||
            @iron-select=${this._setOutputDevice}
 | 
			
		||||
          >
 | 
			
		||||
            <paper-listbox
 | 
			
		||||
@@ -81,26 +88,26 @@ class HassioAddonAudio extends LitElement {
 | 
			
		||||
              .selected=${this._selectedOutput!}
 | 
			
		||||
            >
 | 
			
		||||
              ${this._outputDevices &&
 | 
			
		||||
              this._outputDevices.map((item) => {
 | 
			
		||||
                return html`
 | 
			
		||||
              this._outputDevices.map(
 | 
			
		||||
                (item) => html`
 | 
			
		||||
                  <paper-item device=${item.device || ""}
 | 
			
		||||
                    >${item.name}</paper-item
 | 
			
		||||
                  >
 | 
			
		||||
                `;
 | 
			
		||||
              })}
 | 
			
		||||
                `
 | 
			
		||||
              )}
 | 
			
		||||
            </paper-listbox>
 | 
			
		||||
          </paper-dropdown-menu>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="card-actions">
 | 
			
		||||
          <ha-progress-button @click=${this._saveSettings}>
 | 
			
		||||
            Save
 | 
			
		||||
            ${this.supervisor.localize("common.save")}
 | 
			
		||||
          </ha-progress-button>
 | 
			
		||||
        </div>
 | 
			
		||||
      </ha-card>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult[] {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return [
 | 
			
		||||
      haStyle,
 | 
			
		||||
      hassioStyle,
 | 
			
		||||
@@ -152,7 +159,7 @@ class HassioAddonAudio extends LitElement {
 | 
			
		||||
 | 
			
		||||
    const noDevice: HassioHardwareAudioDevice = {
 | 
			
		||||
      device: "default",
 | 
			
		||||
      name: "Default",
 | 
			
		||||
      name: this.supervisor.localize("addon.configuration.audio.default"),
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    try {
 | 
			
		||||
@@ -189,7 +196,7 @@ class HassioAddonAudio extends LitElement {
 | 
			
		||||
    try {
 | 
			
		||||
      await setHassioAddonOption(this.hass, this.addon.slug, data);
 | 
			
		||||
      if (this.addon?.state === "started") {
 | 
			
		||||
        await suggestAddonRestart(this, this.hass, this.addon);
 | 
			
		||||
        await suggestAddonRestart(this, this.hass, this.supervisor, this.addon);
 | 
			
		||||
      }
 | 
			
		||||
    } catch {
 | 
			
		||||
      this._error = "Failed to set addon audio device";
 | 
			
		||||
 
 | 
			
		||||
@@ -1,17 +1,11 @@
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  html,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
 | 
			
		||||
import { customElement, property } from "lit/decorators";
 | 
			
		||||
import "../../../../src/components/ha-circular-progress";
 | 
			
		||||
import { HassioAddonDetails } from "../../../../src/data/hassio/addon";
 | 
			
		||||
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
 | 
			
		||||
import { haStyle } from "../../../../src/resources/styles";
 | 
			
		||||
import { HomeAssistant } from "../../../../src/types";
 | 
			
		||||
import { hassioStyle } from "../../resources/hassio-style";
 | 
			
		||||
import "../../../../src/components/ha-circular-progress";
 | 
			
		||||
import "./hassio-addon-audio";
 | 
			
		||||
import "./hassio-addon-config";
 | 
			
		||||
import "./hassio-addon-network";
 | 
			
		||||
@@ -20,39 +14,56 @@ import "./hassio-addon-network";
 | 
			
		||||
class HassioAddonConfigDashboard extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) public hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public supervisor!: Supervisor;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public addon?: HassioAddonDetails;
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    if (!this.addon) {
 | 
			
		||||
      return html`<ha-circular-progress active></ha-circular-progress>`;
 | 
			
		||||
    }
 | 
			
		||||
    const hasConfiguration =
 | 
			
		||||
      (this.addon.options && Object.keys(this.addon.options).length) ||
 | 
			
		||||
      (this.addon.schema && Object.keys(this.addon.schema).length);
 | 
			
		||||
 | 
			
		||||
    return html`
 | 
			
		||||
      <div class="content">
 | 
			
		||||
        <hassio-addon-config
 | 
			
		||||
          .hass=${this.hass}
 | 
			
		||||
          .addon=${this.addon}
 | 
			
		||||
        ></hassio-addon-config>
 | 
			
		||||
        ${this.addon.network
 | 
			
		||||
        ${hasConfiguration || this.addon.network || this.addon.audio
 | 
			
		||||
          ? html`
 | 
			
		||||
              <hassio-addon-network
 | 
			
		||||
                .hass=${this.hass}
 | 
			
		||||
                .addon=${this.addon}
 | 
			
		||||
              ></hassio-addon-network>
 | 
			
		||||
              ${hasConfiguration
 | 
			
		||||
                ? html`
 | 
			
		||||
                    <hassio-addon-config
 | 
			
		||||
                      .hass=${this.hass}
 | 
			
		||||
                      .addon=${this.addon}
 | 
			
		||||
                      .supervisor=${this.supervisor}
 | 
			
		||||
                    ></hassio-addon-config>
 | 
			
		||||
                  `
 | 
			
		||||
                : ""}
 | 
			
		||||
              ${this.addon.network
 | 
			
		||||
                ? html`
 | 
			
		||||
                    <hassio-addon-network
 | 
			
		||||
                      .hass=${this.hass}
 | 
			
		||||
                      .addon=${this.addon}
 | 
			
		||||
                      .supervisor=${this.supervisor}
 | 
			
		||||
                    ></hassio-addon-network>
 | 
			
		||||
                  `
 | 
			
		||||
                : ""}
 | 
			
		||||
              ${this.addon.audio
 | 
			
		||||
                ? html`
 | 
			
		||||
                    <hassio-addon-audio
 | 
			
		||||
                      .hass=${this.hass}
 | 
			
		||||
                      .addon=${this.addon}
 | 
			
		||||
                      .supervisor=${this.supervisor}
 | 
			
		||||
                    ></hassio-addon-audio>
 | 
			
		||||
                  `
 | 
			
		||||
                : ""}
 | 
			
		||||
            `
 | 
			
		||||
          : ""}
 | 
			
		||||
        ${this.addon.audio
 | 
			
		||||
          ? html`
 | 
			
		||||
              <hassio-addon-audio
 | 
			
		||||
                .hass=${this.hass}
 | 
			
		||||
                .addon=${this.addon}
 | 
			
		||||
              ></hassio-addon-audio>
 | 
			
		||||
            `
 | 
			
		||||
          : ""}
 | 
			
		||||
          : this.supervisor.localize("addon.configuration.no_configuration")}
 | 
			
		||||
      </div>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult[] {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return [
 | 
			
		||||
      haStyle,
 | 
			
		||||
      hassioStyle,
 | 
			
		||||
 
 | 
			
		||||
@@ -1,84 +1,231 @@
 | 
			
		||||
import "@material/mwc-button";
 | 
			
		||||
import { ActionDetail } from "@material/mwc-list";
 | 
			
		||||
import "@material/mwc-list/mwc-list-item";
 | 
			
		||||
import { mdiDotsVertical } from "@mdi/js";
 | 
			
		||||
import "@polymer/iron-autogrow-textarea/iron-autogrow-textarea";
 | 
			
		||||
import { DEFAULT_SCHEMA, Type } from "js-yaml";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  CSSResultGroup,
 | 
			
		||||
  html,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  PropertyValues,
 | 
			
		||||
  query,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
} from "lit";
 | 
			
		||||
import { customElement, property, query, state } from "lit/decorators";
 | 
			
		||||
import memoizeOne from "memoize-one";
 | 
			
		||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
 | 
			
		||||
import "../../../../src/components/buttons/ha-progress-button";
 | 
			
		||||
import "../../../../src/components/ha-button-menu";
 | 
			
		||||
import "../../../../src/components/ha-card";
 | 
			
		||||
import "../../../../src/components/ha-form/ha-form";
 | 
			
		||||
import type { HaFormSchema } from "../../../../src/components/ha-form/ha-form";
 | 
			
		||||
import "../../../../src/components/ha-formfield";
 | 
			
		||||
import "../../../../src/components/ha-switch";
 | 
			
		||||
import "../../../../src/components/ha-yaml-editor";
 | 
			
		||||
import type { HaYamlEditor } from "../../../../src/components/ha-yaml-editor";
 | 
			
		||||
import {
 | 
			
		||||
  HassioAddonDetails,
 | 
			
		||||
  HassioAddonSetOptionParams,
 | 
			
		||||
  setHassioAddonOption,
 | 
			
		||||
  validateHassioAddonOption,
 | 
			
		||||
} from "../../../../src/data/hassio/addon";
 | 
			
		||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
 | 
			
		||||
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
 | 
			
		||||
import { showConfirmationDialog } from "../../../../src/dialogs/generic/show-dialog-box";
 | 
			
		||||
import { haStyle } from "../../../../src/resources/styles";
 | 
			
		||||
import type { HomeAssistant } from "../../../../src/types";
 | 
			
		||||
import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart";
 | 
			
		||||
import { hassioStyle } from "../../resources/hassio-style";
 | 
			
		||||
 | 
			
		||||
const SUPPORTED_UI_TYPES = ["string", "select", "boolean", "integer", "float"];
 | 
			
		||||
 | 
			
		||||
const ADDON_YAML_SCHEMA = DEFAULT_SCHEMA.extend([
 | 
			
		||||
  new Type("!secret", {
 | 
			
		||||
    kind: "scalar",
 | 
			
		||||
    construct: (data) => `!secret ${data}`,
 | 
			
		||||
  }),
 | 
			
		||||
]);
 | 
			
		||||
 | 
			
		||||
@customElement("hassio-addon-config")
 | 
			
		||||
class HassioAddonConfig extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) public hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public addon!: HassioAddonDetails;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _error?: string;
 | 
			
		||||
  @property({ attribute: false }) public hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public supervisor!: Supervisor;
 | 
			
		||||
 | 
			
		||||
  @property({ type: Boolean }) private _configHasChanged = false;
 | 
			
		||||
 | 
			
		||||
  @property({ type: Boolean }) private _valid = true;
 | 
			
		||||
 | 
			
		||||
  @query("ha-yaml-editor", true) private _editor!: HaYamlEditor;
 | 
			
		||||
  @state() private _canShowSchema = false;
 | 
			
		||||
 | 
			
		||||
  @state() private _showOptional = false;
 | 
			
		||||
 | 
			
		||||
  @state() private _error?: string;
 | 
			
		||||
 | 
			
		||||
  @state() private _options?: Record<string, unknown>;
 | 
			
		||||
 | 
			
		||||
  @state() private _yamlMode = false;
 | 
			
		||||
 | 
			
		||||
  @query("ha-yaml-editor") private _editor?: HaYamlEditor;
 | 
			
		||||
 | 
			
		||||
  public computeLabel = (entry: HaFormSchema): string =>
 | 
			
		||||
    this.addon.translations[this.hass.language]?.configuration?.[entry.name]
 | 
			
		||||
      ?.name ||
 | 
			
		||||
    this.addon.translations.en?.configuration?.[entry.name].name ||
 | 
			
		||||
    entry.name;
 | 
			
		||||
 | 
			
		||||
  private _filteredShchema = memoizeOne(
 | 
			
		||||
    (options: Record<string, unknown>, schema: HaFormSchema[]) =>
 | 
			
		||||
      schema.filter((entry) => entry.name in options || entry.required)
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  protected render(): TemplateResult {
 | 
			
		||||
    const showForm =
 | 
			
		||||
      !this._yamlMode && this._canShowSchema && this.addon.schema;
 | 
			
		||||
    const hasHiddenOptions =
 | 
			
		||||
      showForm &&
 | 
			
		||||
      JSON.stringify(this.addon.schema) !==
 | 
			
		||||
        JSON.stringify(
 | 
			
		||||
          this._filteredShchema(this.addon.options, this.addon.schema!)
 | 
			
		||||
        );
 | 
			
		||||
    return html`
 | 
			
		||||
      <h1>${this.addon.name}</h1>
 | 
			
		||||
      <ha-card header="Configuration">
 | 
			
		||||
        <div class="card-content">
 | 
			
		||||
          <ha-yaml-editor
 | 
			
		||||
            @value-changed=${this._configChanged}
 | 
			
		||||
          ></ha-yaml-editor>
 | 
			
		||||
          ${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
 | 
			
		||||
          ${this._valid ? "" : html` <div class="errors">Invalid YAML</div> `}
 | 
			
		||||
      <ha-card>
 | 
			
		||||
        <div class="header">
 | 
			
		||||
          <h2>
 | 
			
		||||
            ${this.supervisor.localize("addon.configuration.options.header")}
 | 
			
		||||
          </h2>
 | 
			
		||||
          <div class="card-menu">
 | 
			
		||||
            <ha-button-menu corner="BOTTOM_START" @action=${this._handleAction}>
 | 
			
		||||
              <mwc-icon-button slot="trigger">
 | 
			
		||||
                <ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon>
 | 
			
		||||
              </mwc-icon-button>
 | 
			
		||||
              <mwc-list-item .disabled=${!this._canShowSchema}>
 | 
			
		||||
                ${this._yamlMode
 | 
			
		||||
                  ? this.supervisor.localize(
 | 
			
		||||
                      "addon.configuration.options.edit_in_ui"
 | 
			
		||||
                    )
 | 
			
		||||
                  : this.supervisor.localize(
 | 
			
		||||
                      "addon.configuration.options.edit_in_yaml"
 | 
			
		||||
                    )}
 | 
			
		||||
              </mwc-list-item>
 | 
			
		||||
              <mwc-list-item class="warning">
 | 
			
		||||
                ${this.supervisor.localize("common.reset_defaults")}
 | 
			
		||||
              </mwc-list-item>
 | 
			
		||||
            </ha-button-menu>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="card-actions">
 | 
			
		||||
          <ha-progress-button class="warning" @click=${this._resetTapped}>
 | 
			
		||||
            Reset to defaults
 | 
			
		||||
          </ha-progress-button>
 | 
			
		||||
 | 
			
		||||
        <div class="card-content">
 | 
			
		||||
          ${showForm
 | 
			
		||||
            ? html`<ha-form
 | 
			
		||||
                .data=${this._options!}
 | 
			
		||||
                @value-changed=${this._configChanged}
 | 
			
		||||
                .computeLabel=${this.computeLabel}
 | 
			
		||||
                .schema=${this._showOptional
 | 
			
		||||
                  ? this.addon.schema!
 | 
			
		||||
                  : this._filteredShchema(
 | 
			
		||||
                      this.addon.options,
 | 
			
		||||
                      this.addon.schema!
 | 
			
		||||
                    )}
 | 
			
		||||
              ></ha-form>`
 | 
			
		||||
            : html` <ha-yaml-editor
 | 
			
		||||
                @value-changed=${this._configChanged}
 | 
			
		||||
                .schema=${ADDON_YAML_SCHEMA}
 | 
			
		||||
              ></ha-yaml-editor>`}
 | 
			
		||||
          ${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
 | 
			
		||||
          ${!this._yamlMode ||
 | 
			
		||||
          (this._canShowSchema && this.addon.schema) ||
 | 
			
		||||
          this._valid
 | 
			
		||||
            ? ""
 | 
			
		||||
            : html`
 | 
			
		||||
                <div class="errors">
 | 
			
		||||
                  ${this.supervisor.localize(
 | 
			
		||||
                    "addon.configuration.options.invalid_yaml"
 | 
			
		||||
                  )}
 | 
			
		||||
                </div>
 | 
			
		||||
              `}
 | 
			
		||||
        </div>
 | 
			
		||||
        ${hasHiddenOptions
 | 
			
		||||
          ? html`<ha-formfield
 | 
			
		||||
              class="show-additional"
 | 
			
		||||
              .label=${this.supervisor.localize(
 | 
			
		||||
                "addon.configuration.options.show_unused_optional"
 | 
			
		||||
              )}
 | 
			
		||||
            >
 | 
			
		||||
              <ha-switch
 | 
			
		||||
                @change=${this._toggleOptional}
 | 
			
		||||
                .checked=${this._showOptional}
 | 
			
		||||
              >
 | 
			
		||||
              </ha-switch>
 | 
			
		||||
            </ha-formfield>`
 | 
			
		||||
          : ""}
 | 
			
		||||
        <div class="card-actions right">
 | 
			
		||||
          <ha-progress-button
 | 
			
		||||
            @click=${this._saveTapped}
 | 
			
		||||
            .disabled=${!this._configHasChanged || !this._valid}
 | 
			
		||||
          >
 | 
			
		||||
            Save
 | 
			
		||||
            ${this.supervisor.localize("common.save")}
 | 
			
		||||
          </ha-progress-button>
 | 
			
		||||
        </div>
 | 
			
		||||
      </ha-card>
 | 
			
		||||
    `;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  protected firstUpdated(changedProps) {
 | 
			
		||||
    super.firstUpdated(changedProps);
 | 
			
		||||
    this._canShowSchema = !this.addon.schema!.find(
 | 
			
		||||
      // @ts-ignore
 | 
			
		||||
      (entry) => !SUPPORTED_UI_TYPES.includes(entry.type) || entry.multiple
 | 
			
		||||
    );
 | 
			
		||||
    this._yamlMode = !this._canShowSchema;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  protected updated(changedProperties: PropertyValues): void {
 | 
			
		||||
    super.updated(changedProperties);
 | 
			
		||||
    if (changedProperties.has("addon")) {
 | 
			
		||||
      this._editor.setValue(this.addon.options);
 | 
			
		||||
      this._options = { ...this.addon.options };
 | 
			
		||||
    }
 | 
			
		||||
    super.updated(changedProperties);
 | 
			
		||||
    if (
 | 
			
		||||
      changedProperties.has("_yamlMode") ||
 | 
			
		||||
      changedProperties.has("_options")
 | 
			
		||||
    ) {
 | 
			
		||||
      if (this._yamlMode) {
 | 
			
		||||
        const editor = this._editor;
 | 
			
		||||
        if (editor) {
 | 
			
		||||
          editor.setValue(this._options!);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private _handleAction(ev: CustomEvent<ActionDetail>) {
 | 
			
		||||
    switch (ev.detail.index) {
 | 
			
		||||
      case 0:
 | 
			
		||||
        this._yamlMode = !this._yamlMode;
 | 
			
		||||
        break;
 | 
			
		||||
      case 1:
 | 
			
		||||
        this._resetTapped(ev);
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private _toggleOptional() {
 | 
			
		||||
    this._showOptional = !this._showOptional;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private _configChanged(ev): void {
 | 
			
		||||
    this._configHasChanged = true;
 | 
			
		||||
    this._valid = ev.detail.isValid;
 | 
			
		||||
    if (this.addon.schema && this._canShowSchema && !this._yamlMode) {
 | 
			
		||||
      this._valid = true;
 | 
			
		||||
      this._configHasChanged = true;
 | 
			
		||||
      this._options! = ev.detail.value;
 | 
			
		||||
    } else {
 | 
			
		||||
      this._configHasChanged = true;
 | 
			
		||||
      this._valid = ev.detail.isValid;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _resetTapped(ev: CustomEvent): Promise<void> {
 | 
			
		||||
@@ -86,10 +233,10 @@ class HassioAddonConfig extends LitElement {
 | 
			
		||||
    button.progress = true;
 | 
			
		||||
 | 
			
		||||
    const confirmed = await showConfirmationDialog(this, {
 | 
			
		||||
      title: this.addon.name,
 | 
			
		||||
      text: "Are you sure you want to reset all your options?",
 | 
			
		||||
      confirmText: "reset options",
 | 
			
		||||
      dismissText: "no",
 | 
			
		||||
      title: this.supervisor.localize("confirm.reset_options.title"),
 | 
			
		||||
      text: this.supervisor.localize("confirm.reset_options.text"),
 | 
			
		||||
      confirmText: this.supervisor.localize("common.reset_options"),
 | 
			
		||||
      dismissText: this.supervisor.localize("common.cancel"),
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    if (!confirmed) {
 | 
			
		||||
@@ -111,48 +258,56 @@ class HassioAddonConfig extends LitElement {
 | 
			
		||||
      };
 | 
			
		||||
      fireEvent(this, "hass-api-called", eventdata);
 | 
			
		||||
    } catch (err) {
 | 
			
		||||
      this._error = `Failed to reset addon configuration, ${extractApiErrorMessage(
 | 
			
		||||
        err
 | 
			
		||||
      )}`;
 | 
			
		||||
      this._error = this.supervisor.localize(
 | 
			
		||||
        "addon.common.update_available",
 | 
			
		||||
        "error",
 | 
			
		||||
        extractApiErrorMessage(err)
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
    button.progress = false;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private async _saveTapped(ev: CustomEvent): Promise<void> {
 | 
			
		||||
    const button = ev.currentTarget as any;
 | 
			
		||||
    const eventdata = {
 | 
			
		||||
      success: true,
 | 
			
		||||
      response: undefined,
 | 
			
		||||
      path: "options",
 | 
			
		||||
    };
 | 
			
		||||
    button.progress = true;
 | 
			
		||||
 | 
			
		||||
    let data: HassioAddonSetOptionParams;
 | 
			
		||||
    this._error = undefined;
 | 
			
		||||
 | 
			
		||||
    try {
 | 
			
		||||
      data = {
 | 
			
		||||
        options: this._editor.value,
 | 
			
		||||
      };
 | 
			
		||||
    } catch (err) {
 | 
			
		||||
      this._error = err;
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    try {
 | 
			
		||||
      await setHassioAddonOption(this.hass, this.addon.slug, data);
 | 
			
		||||
      const validation = await validateHassioAddonOption(
 | 
			
		||||
        this.hass,
 | 
			
		||||
        this.addon.slug,
 | 
			
		||||
        this._editor?.value
 | 
			
		||||
      );
 | 
			
		||||
      if (!validation.valid) {
 | 
			
		||||
        throw Error(validation.message);
 | 
			
		||||
      }
 | 
			
		||||
      await setHassioAddonOption(this.hass, this.addon.slug, {
 | 
			
		||||
        options: this._yamlMode ? this._editor?.value : this._options,
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      this._configHasChanged = false;
 | 
			
		||||
      const eventdata = {
 | 
			
		||||
        success: true,
 | 
			
		||||
        response: undefined,
 | 
			
		||||
        path: "options",
 | 
			
		||||
      };
 | 
			
		||||
      fireEvent(this, "hass-api-called", eventdata);
 | 
			
		||||
      if (this.addon?.state === "started") {
 | 
			
		||||
        await suggestAddonRestart(this, this.hass, this.addon);
 | 
			
		||||
        await suggestAddonRestart(this, this.hass, this.supervisor, this.addon);
 | 
			
		||||
      }
 | 
			
		||||
    } catch (err) {
 | 
			
		||||
      this._error = `Failed to save addon configuration, ${extractApiErrorMessage(
 | 
			
		||||
        err
 | 
			
		||||
      )}`;
 | 
			
		||||
      this._error = this.supervisor.localize(
 | 
			
		||||
        "addon.failed_to_save",
 | 
			
		||||
        "error",
 | 
			
		||||
        extractApiErrorMessage(err)
 | 
			
		||||
      );
 | 
			
		||||
      eventdata.success = false;
 | 
			
		||||
    }
 | 
			
		||||
    button.progress = false;
 | 
			
		||||
    fireEvent(this, "hass-api-called", eventdata);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult[] {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return [
 | 
			
		||||
      haStyle,
 | 
			
		||||
      hassioStyle,
 | 
			
		||||
@@ -178,6 +333,36 @@ class HassioAddonConfig extends LitElement {
 | 
			
		||||
        .syntaxerror {
 | 
			
		||||
          color: var(--error-color);
 | 
			
		||||
        }
 | 
			
		||||
        .card-menu {
 | 
			
		||||
          float: right;
 | 
			
		||||
          z-index: 3;
 | 
			
		||||
          --mdc-theme-text-primary-on-background: var(--primary-text-color);
 | 
			
		||||
        }
 | 
			
		||||
        mwc-list-item[disabled] {
 | 
			
		||||
          --mdc-theme-text-primary-on-background: var(--disabled-text-color);
 | 
			
		||||
        }
 | 
			
		||||
        .header {
 | 
			
		||||
          display: flex;
 | 
			
		||||
          justify-content: space-between;
 | 
			
		||||
        }
 | 
			
		||||
        .header h2 {
 | 
			
		||||
          color: var(--ha-card-header-color, --primary-text-color);
 | 
			
		||||
          font-family: var(--ha-card-header-font-family, inherit);
 | 
			
		||||
          font-size: var(--ha-card-header-font-size, 24px);
 | 
			
		||||
          letter-spacing: -0.012em;
 | 
			
		||||
          line-height: 48px;
 | 
			
		||||
          padding: 12px 16px 16px;
 | 
			
		||||
          display: block;
 | 
			
		||||
          margin-block: 0px;
 | 
			
		||||
          font-weight: normal;
 | 
			
		||||
        }
 | 
			
		||||
        .card-actions.right {
 | 
			
		||||
          justify-content: flex-end;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        .show-additional {
 | 
			
		||||
          padding: 16px;
 | 
			
		||||
        }
 | 
			
		||||
      `,
 | 
			
		||||
    ];
 | 
			
		||||
  }
 | 
			
		||||
 
 | 
			
		||||
@@ -1,15 +1,13 @@
 | 
			
		||||
import { PaperInputElement } from "@polymer/paper-input/paper-input";
 | 
			
		||||
import {
 | 
			
		||||
  css,
 | 
			
		||||
  CSSResult,
 | 
			
		||||
  customElement,
 | 
			
		||||
  CSSResultGroup,
 | 
			
		||||
  html,
 | 
			
		||||
  internalProperty,
 | 
			
		||||
  LitElement,
 | 
			
		||||
  property,
 | 
			
		||||
  PropertyValues,
 | 
			
		||||
  TemplateResult,
 | 
			
		||||
} from "lit-element";
 | 
			
		||||
} from "lit";
 | 
			
		||||
import { customElement, property, state } from "lit/decorators";
 | 
			
		||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
 | 
			
		||||
import "../../../../src/components/buttons/ha-progress-button";
 | 
			
		||||
import "../../../../src/components/ha-card";
 | 
			
		||||
@@ -19,6 +17,7 @@ import {
 | 
			
		||||
  setHassioAddonOption,
 | 
			
		||||
} from "../../../../src/data/hassio/addon";
 | 
			
		||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
 | 
			
		||||
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
 | 
			
		||||
import { haStyle } from "../../../../src/resources/styles";
 | 
			
		||||
import { HomeAssistant } from "../../../../src/types";
 | 
			
		||||
import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart";
 | 
			
		||||
@@ -38,11 +37,13 @@ interface NetworkItemInput extends PaperInputElement {
 | 
			
		||||
class HassioAddonNetwork extends LitElement {
 | 
			
		||||
  @property({ attribute: false }) public hass!: HomeAssistant;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public supervisor!: Supervisor;
 | 
			
		||||
 | 
			
		||||
  @property({ attribute: false }) public addon!: HassioAddonDetails;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _error?: string;
 | 
			
		||||
  @state() private _error?: string;
 | 
			
		||||
 | 
			
		||||
  @internalProperty() private _config?: NetworkItem[];
 | 
			
		||||
  @state() private _config?: NetworkItem[];
 | 
			
		||||
 | 
			
		||||
  public connectedCallback(): void {
 | 
			
		||||
    super.connectedCallback();
 | 
			
		||||
@@ -55,43 +56,57 @@ class HassioAddonNetwork extends LitElement {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return html`
 | 
			
		||||
      <ha-card header="Network">
 | 
			
		||||
      <ha-card
 | 
			
		||||
        .header=${this.supervisor.localize(
 | 
			
		||||
          "addon.configuration.network.header"
 | 
			
		||||
        )}
 | 
			
		||||
      >
 | 
			
		||||
        <div class="card-content">
 | 
			
		||||
          ${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
 | 
			
		||||
 | 
			
		||||
          <table>
 | 
			
		||||
            <tbody>
 | 
			
		||||
              <tr>
 | 
			
		||||
                <th>Container</th>
 | 
			
		||||
                <th>Host</th>
 | 
			
		||||
                <th>Description</th>
 | 
			
		||||
                <th>
 | 
			
		||||
                  ${this.supervisor.localize(
 | 
			
		||||
                    "addon.configuration.network.container"
 | 
			
		||||
                  )}
 | 
			
		||||
                </th>
 | 
			
		||||
                <th>
 | 
			
		||||
                  ${this.supervisor.localize(
 | 
			
		||||
                    "addon.configuration.network.host"
 | 
			
		||||
                  )}
 | 
			
		||||
                </th>
 | 
			
		||||
                <th>${this.supervisor.localize("common.description")}</th>
 | 
			
		||||
              </tr>
 | 
			
		||||
              ${this._config!.map((item) => {
 | 
			
		||||
                return html`
 | 
			
		||||
              ${this._config!.map(
 | 
			
		||||
                (item) => html`
 | 
			
		||||
                  <tr>
 | 
			
		||||
                    <td>${item.container}</td>
 | 
			
		||||
                    <td>
 | 
			
		||||
                      <paper-input
 | 
			
		||||
                        @value-changed=${this._configChanged}
 | 
			
		||||
                        placeholder="disabled"
 | 
			
		||||
                        placeholder="${this.supervisor.localize(
 | 
			
		||||
                          "addon.configuration.network.disabled"
 | 
			
		||||
                        )}"
 | 
			
		||||
                        .value=${item.host ? String(item.host) : ""}
 | 
			
		||||
                        .container=${item.container}
 | 
			
		||||
                        no-label-float
 | 
			
		||||
                      ></paper-input>
 | 
			
		||||
                    </td>
 | 
			
		||||
                    <td>${item.description}</td>
 | 
			
		||||
                    <td>${this._computeDescription(item)}</td>
 | 
			
		||||
                  </tr>
 | 
			
		||||
                `;
 | 
			
		||||
              })}
 | 
			
		||||
                `
 | 
			
		||||
              )}
 | 
			
		||||
            </tbody>
 | 
			
		||||
          </table>
 | 
			
		||||
        </div>
 | 
			
		||||
        <div class="card-actions">
 | 
			
		||||
          <ha-progress-button class="warning" @click=${this._resetTapped}>
 | 
			
		||||
            Reset to defaults
 | 
			
		||||
            ${this.supervisor.localize("common.reset_defaults")}
 | 
			
		||||
          </ha-progress-button>
 | 
			
		||||
          <ha-progress-button @click=${this._saveTapped}>
 | 
			
		||||
            Save
 | 
			
		||||
            ${this.supervisor.localize("common.save")}
 | 
			
		||||
          </ha-progress-button>
 | 
			
		||||
        </div>
 | 
			
		||||
      </ha-card>
 | 
			
		||||
@@ -105,16 +120,20 @@ class HassioAddonNetwork extends LitElement {
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  private _computeDescription = (item: NetworkItem): string =>
 | 
			
		||||
    this.addon.translations[this.hass.language]?.network?.[item.container]
 | 
			
		||||
      ?.description ||
 | 
			
		||||
    this.addon.translations.en?.network?.[item.container]?.description ||
 | 
			
		||||
    item.description;
 | 
			
		||||
 | 
			
		||||
  private _setNetworkConfig(): void {
 | 
			
		||||
    const network = this.addon.network || {};
 | 
			
		||||
    const description = this.addon.network_description || {};
 | 
			
		||||
    const items: NetworkItem[] = Object.keys(network).map((key) => {
 | 
			
		||||
      return {
 | 
			
		||||
        container: key,
 | 
			
		||||
        host: network[key],
 | 
			
		||||
        description: description[key],
 | 
			
		||||
      };
 | 
			
		||||
    });
 | 
			
		||||
    const items: NetworkItem[] = Object.keys(network).map((key) => ({
 | 
			
		||||
      container: key,
 | 
			
		||||
      host: network[key],
 | 
			
		||||
      description: description[key],
 | 
			
		||||
    }));
 | 
			
		||||
    this._config = items.sort((a, b) => (a.container > b.container ? 1 : -1));
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
@@ -147,12 +166,14 @@ class HassioAddonNetwork extends LitElement {
 | 
			
		||||
      };
 | 
			
		||||
      fireEvent(this, "hass-api-called", eventdata);
 | 
			
		||||
      if (this.addon?.state === "started") {
 | 
			
		||||
        await suggestAddonRestart(this, this.hass, this.addon);
 | 
			
		||||
        await suggestAddonRestart(this, this.hass, this.supervisor, this.addon);
 | 
			
		||||
      }
 | 
			
		||||
    } catch (err) {
 | 
			
		||||
      this._error = `Failed to set addon network configuration, ${extractApiErrorMessage(
 | 
			
		||||
        err
 | 
			
		||||
      )}`;
 | 
			
		||||
      this._error = this.supervisor.localize(
 | 
			
		||||
        "addon.failed_to_reset",
 | 
			
		||||
        "error",
 | 
			
		||||
        extractApiErrorMessage(err)
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    button.progress = false;
 | 
			
		||||
@@ -181,17 +202,19 @@ class HassioAddonNetwork extends LitElement {
 | 
			
		||||
      };
 | 
			
		||||
      fireEvent(this, "hass-api-called", eventdata);
 | 
			
		||||
      if (this.addon?.state === "started") {
 | 
			
		||||
        await suggestAddonRestart(this, this.hass, this.addon);
 | 
			
		||||
        await suggestAddonRestart(this, this.hass, this.supervisor, this.addon);
 | 
			
		||||
      }
 | 
			
		||||
    } catch (err) {
 | 
			
		||||
      this._error = `Failed to set addon network configuration, ${extractApiErrorMessage(
 | 
			
		||||
        err
 | 
			
		||||
      )}`;
 | 
			
		||||
      this._error = this.supervisor.localize(
 | 
			
		||||
        "addon.failed_to_save",
 | 
			
		||||
        "error",
 | 
			
		||||
        extractApiErrorMessage(err)
 | 
			
		||||
      );
 | 
			
		||||
    }
 | 
			
		||||
    button.progress = false;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  static get styles(): CSSResult[] {
 | 
			
		||||
  static get styles(): CSSResultGroup {
 | 
			
		||||
    return [
 | 
			
		||||
      haStyle,
 | 
			
		||||
      hassioStyle,
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user