mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-22 18:19:48 +00:00 
			
		
		
		
	Compare commits
	
		
			1450 Commits
		
	
	
		
			template-e
			...
			data-disk-
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 2c440976aa | ||
|   | abbfe7200a | ||
|   | 419942112b | ||
|   | 597d4a0426 | ||
|   | e023d60be7 | ||
|   | 41a7b42037 | ||
|   | 2936865c55 | ||
|   | ff2bf1f3c1 | ||
|   | 1bccbd4173 | ||
|   | d7f00df391 | ||
|   | 22f88c59c7 | ||
|   | 8721776839 | ||
|   | a89da0dac0 | ||
|   | e4b4dc4ae9 | ||
|   | b26c44b2b9 | ||
|   | 68095417b9 | ||
|   | b6344eb6e8 | ||
|   | 224302cfef | ||
|   | abc4816888 | ||
|   | 21e14bd644 | ||
|   | a89caccd32 | ||
|   | 03dc3e52b7 | ||
|   | f04be8efa6 | ||
|   | 2c32f6bcb3 | ||
|   | a3a08ff5c7 | ||
|   | ea51186767 | ||
|   | 49494c572b | ||
|   | fcac3fa164 | ||
|   | 9c1153ef37 | ||
|   | 0adc4b33ef | ||
|   | c0f3215340 | ||
|   | bab1e6a95f | ||
|   | 53b26a43c0 | ||
|   | 2240d019f5 | ||
|   | cb11c6b3ea | ||
|   | 5893559951 | ||
|   | 8408d25cef | ||
|   | 1ac2ffcf02 | ||
|   | 6b6c38c2c8 | ||
|   | e55df73a91 | ||
|   | 360c2cbfa3 | ||
|   | aba96674f3 | ||
|   | 5c3d85fc90 | ||
|   | 6486b7fd4c | ||
|   | 5f3e980de0 | ||
|   | d0edbec5fb | ||
|   | 5d46963e8a | ||
|   | 321f441b63 | ||
|   | d55bade070 | ||
|   | 6ba6b821f5 | ||
|   | b3dedae115 | ||
|   | 5a1070c30f | ||
|   | 40664997e1 | ||
|   | c6e83cb7c0 | ||
|   | e7e27e794c | ||
|   | 1073dbe6ab | ||
|   | 2bd9b5a015 | ||
|   | bc09febd2c | ||
|   | b2a87c90a2 | ||
|   | d6dbbcb0de | ||
|   | 9ccb5360b3 | ||
|   | 0187c4faff | ||
|   | 605172a0bc | ||
|   | 8565a0d911 | ||
|   | 61c8d23a7e | ||
|   | 5e3487ed59 | ||
|   | d5a161769c | ||
|   | 1692f9c2dd | ||
|   | 0cbac8bb44 | ||
|   | 35a81e7f11 | ||
|   | ac64d293e7 | ||
|   | 708b8787c5 | ||
|   | 2bddd151eb | ||
|   | 43a585187c | ||
|   | 324658a36b | ||
|   | dd9a9b34d1 | ||
|   | 2ab0e40952 | ||
|   | dfea80ae96 | ||
|   | 6e38f5accf | ||
|   | 7c952d92bf | ||
|   | 2fae0d2d95 | ||
|   | 67ab63f00e | ||
|   | 719f9c28af | ||
|   | 035d621109 | ||
|   | 791f3b896d | ||
|   | fe2172a660 | ||
|   | 640fbd616b | ||
|   | 900efe8a36 | ||
|   | 5bd92d04d9 | ||
|   | b15684bcbd | ||
|   | a93222dbb2 | ||
|   | 20744e90a0 | ||
|   | 32777b4259 | ||
|   | 271120999c | ||
|   | 68fe13a67d | ||
|   | f3606014c6 | ||
|   | efbf4482b2 | ||
|   | 21a3b4f8e2 | ||
|   | de23b2d046 | ||
|   | bd8f436c1d | ||
|   | e963735dba | ||
|   | 46c981103d | ||
|   | f6d02d8fc6 | ||
|   | e08f691510 | ||
|   | af9199aaff | ||
|   | 8576b13f74 | ||
|   | 2270d8a795 | ||
|   | f4dcce6d6c | ||
|   | b802a410b9 | ||
|   | 9e3d339ec5 | ||
|   | fb97a98b97 | ||
|   | 72773f3bc8 | ||
|   | b0fd93e0c3 | ||
|   | 7aa2ec78f2 | ||
|   | 047e856a61 | ||
|   | dbe209e3f2 | ||
|   | e0d23ee6cf | ||
|   | 7a35f46370 | ||
|   | 4a4465efb6 | ||
|   | 3a112531cc | ||
|   | 456209dded | ||
|   | 2556b0d157 | ||
|   | 1e8903fd76 | ||
|   | ad9f18c231 | ||
|   | 63e3de00cb | ||
|   | d06ffeeede | ||
|   | 3479fb9d94 | ||
|   | 304bd002ae | ||
|   | 5dad18c85f | ||
|   | 19e4c0657a | ||
|   | 44548fdc33 | ||
|   | d8929074b5 | ||
|   | e11532ae92 | ||
|   | eff48acdf4 | ||
|   | a9850f9641 | ||
|   | aab0b8a3ce | ||
|   | b12e062d94 | ||
|   | b36e342f15 | ||
|   | f686816c86 | ||
|   | dc50e54afc | ||
|   | 3897e3d452 | ||
|   | 2557b03b11 | ||
|   | 29d29a337f | ||
|   | 34f8e5e28d | ||
|   | afd1a68c62 | ||
|   | dbcf1cb907 | ||
|   | 9ca64f9789 | ||
|   | 4c247ac49d | ||
|   | e8a406526b | ||
|   | 7fcea16c6b | ||
|   | 028b799d2c | ||
|   | 3485296e23 | ||
|   | 03078cdd45 | ||
|   | 740310800d | ||
|   | 6d7c558482 | ||
|   | fdb10515c3 | ||
|   | 5a0f13c485 | ||
|   | 80b330ad7b | ||
|   | d929e1c134 | ||
|   | 5e40dcdc38 | ||
|   | 1dd3e2a83b | ||
|   | a62742fad9 | ||
|   | 1f9c45b11c | ||
|   | 68bec5e158 | ||
|   | 37f1bd7d63 | ||
|   | 5ba24211e2 | ||
|   | d699647418 | ||
|   | b246502cb6 | ||
|   | 9b33ead8aa | ||
|   | 32e8c1dc6d | ||
|   | e09ef7862e | ||
|   | 85420304d0 | ||
|   | 1c097a669d | ||
|   | 4e1497c5da | ||
|   | 49d426675f | ||
|   | dc6ac668b4 | ||
|   | 4ee24b0845 | ||
|   | ba20aef206 | ||
|   | 41ef6133c1 | ||
|   | 50bd5ee8f7 | ||
|   | 285f3fe330 | ||
|   | 4d01199986 | ||
|   | bcc0052fe0 | ||
|   | 4b592d81bd | ||
|   | 884e323288 | ||
|   | 78b799dd05 | ||
|   | 847fa2e700 | ||
|   | 481a79e311 | ||
|   | f19f2ff321 | ||
|   | 6dc4d7bb70 | ||
|   | 83460a34f4 | ||
|   | 2adbb47373 | ||
|   | 2159a5419a | ||
|   | 044d6a15d9 | ||
|   | b6055062c6 | ||
|   | 6fd85e043b | ||
|   | e07ac52356 | ||
|   | 0f16ba9325 | ||
|   | 378e6d28bc | ||
|   | 539d2b880c | ||
|   | 2982adbfa7 | ||
|   | 5147dff670 | ||
|   | 2cdf78c504 | ||
|   | cfad45b7c2 | ||
|   | 5234e9bce5 | ||
|   | 0ed5454d02 | ||
|   | 03080973be | ||
|   | a4f51b0cb3 | ||
|   | 749079c1c3 | ||
|   | ae10ff42e1 | ||
|   | d4cbdab4a3 | ||
|   | 1bd6392a4c | ||
|   | 1531e99528 | ||
|   | 6e7af18494 | ||
|   | e1bae65aeb | ||
|   | 65bfdf94c9 | ||
|   | 9a92825954 | ||
|   | 469faf509b | ||
|   | f87d4a5ab6 | ||
|   | 9c31b749d7 | ||
|   | 521d5df064 | ||
|   | 4d330fba8a | ||
|   | 2e04a55d5c | ||
|   | 0d9d0aa18b | ||
|   | f8dd1795bc | ||
|   | 1d7007584c | ||
|   | 8cd9f891fb | ||
|   | 6ab0f1db57 | ||
|   | 37d754d069 | ||
|   | e12b194d41 | ||
|   | 07d7fa26fe | ||
|   | 73b9b87ef3 | ||
|   | 0c0091375c | ||
|   | 21a29ed3a5 | ||
|   | a6312f4279 | ||
|   | f459abdf85 | ||
|   | 586fa1d0f0 | ||
|   | bf4192a1f0 | ||
|   | ac31eedf65 | ||
|   | b05dc5141c | ||
|   | 32c6fb14dd | ||
|   | 982c940381 | ||
|   | a7a8aaa887 | ||
|   | bf83a9980e | ||
|   | 11be603ed3 | ||
|   | a432cf8405 | ||
|   | 9dd6b3b72d | ||
|   | faca62b55f | ||
|   | a339de89f5 | ||
|   | e40c90e9c0 | ||
|   | 3f447bb8a7 | ||
|   | 21dca3fbf8 | ||
|   | 1078bb4287 | ||
|   | daeed06e70 | ||
|   | 1206e2d75f | ||
|   | cc81239b9d | ||
|   | e797c01761 | ||
|   | 12f7366968 | ||
|   | b7fd7abe85 | ||
|   | a5e1f3d165 | ||
|   | 212d047ada | ||
|   | 2e16127fde | ||
|   | e8b53a619d | ||
|   | df1ca1fd96 | ||
|   | 8f85132d48 | ||
|   | 349144599c | ||
|   | 979093923b | ||
|   | 137f8ad4cb | ||
|   | c1f462b8f8 | ||
|   | 6701c4c371 | ||
|   | fc6e459c09 | ||
|   | 9e28b3447e | ||
|   | 5c737e1969 | ||
|   | 569765e77e | ||
|   | bc0d63ed12 | ||
|   | 02f9893522 | ||
|   | b4bbe63f0f | ||
|   | fabbcac99f | ||
|   | b1b5ab6949 | ||
|   | 4b9487183b | ||
|   | de5a817953 | ||
|   | 4970f640fa | ||
|   | 18996535b7 | ||
|   | 2a1e31b5e9 | ||
|   | 8ca9a0f409 | ||
|   | fcc89a67ba | ||
|   | 1f377d43c5 | ||
|   | 30d6c68908 | ||
|   | dc781da93a | ||
|   | 36c20e4348 | ||
|   | 4466950bb8 | ||
|   | be29828454 | ||
|   | 7bab245073 | ||
|   | f5dcf0b760 | ||
|   | 8141f78a92 | ||
|   | be244b3d00 | ||
|   | 805f5ff9b6 | ||
|   | 76daeb7e55 | ||
|   | 9594c8106e | ||
|   | ed4052365c | ||
|   | 377ebadc10 | ||
|   | ed4809b71e | ||
|   | db37dffdbb | ||
|   | 13cab7e301 | ||
|   | 0a50fc66e5 | ||
|   | a3d1a3566d | ||
|   | ba0be927ed | ||
|   | 4260606267 | ||
|   | 4665db4f27 | ||
|   | 43503ba085 | ||
|   | 0a83a704f1 | ||
|   | 08de941c90 | ||
|   | 62228ef144 | ||
|   | 9731257782 | ||
|   | 4ec9c9c16e | ||
|   | 45436731e2 | ||
|   | 27730e65e7 | ||
|   | a4aba93d57 | ||
|   | d93db16963 | ||
|   | c327fe11b8 | ||
|   | 4fbc31d0b0 | ||
|   | 9a4a1cb4ec | ||
|   | 202d6957bc | ||
|   | 14fcff7774 | ||
|   | 2c9aa1cab4 | ||
|   | 7745c10d07 | ||
|   | c1d571de42 | ||
|   | cecb66451c | ||
|   | 0ef3421fa2 | ||
|   | f88e238d41 | ||
|   | ce3c8f264d | ||
|   | 9fbd594f37 | ||
|   | 5ad95cad90 | ||
|   | 7e507b40c4 | ||
|   | 446a9b5c02 | ||
|   | e02e61384e | ||
|   | 5deb570fdf | ||
|   | 915c46f144 | ||
|   | 30d6c5eaf3 | ||
|   | 6e50d1166a | ||
|   | 0e3eed0563 | ||
|   | 1b1676cecc | ||
|   | d911fe6a0b | ||
|   | 22253a3385 | ||
|   | 38640c99e3 | ||
|   | d6df8bddea | ||
|   | ddfc4bd98e | ||
|   | 3d6674325c | ||
|   | 194829f5b1 | ||
|   | 11a77253f4 | ||
|   | 67be2343f8 | ||
|   | e9b1b3d853 | ||
|   | 8a33d174d7 | ||
|   | 226d6216b7 | ||
|   | 1925bb01be | ||
|   | 82a4806e01 | ||
|   | ce419fae7b | ||
|   | c68b76e2da | ||
|   | 342020b420 | ||
|   | 1e6e99e3c7 | ||
|   | 2e9aafc377 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 299c863f49 | ||
|   | c2792a28ba | ||
|   | 635a027a8e | ||
|   | a45b8ca8e7 | ||
|   | 1e6e945a07 | ||
|   | 8666e6baae | ||
|   | f71157c24d | ||
|   | e87a2b36cf | ||
|   | 5418474f64 | ||
|   | 8836ba6ceb | ||
|   | 509c5b497a | ||
|   | e00bcc9f48 | ||
|   | bdef9fd040 | ||
|   | c956491ec5 | ||
|   | 68bc549d6a | ||
|   | 9c64eafc21 | ||
|   | b05e86d442 | ||
|   | fe5f9576c6 | ||
|   | 1b282b65b7 | ||
|   | e49664bad3 | ||
|   | 2db9f33c41 | ||
|   | 2a30b55a43 | ||
|   | 9d0b20adce | ||
|   | acd5e1c081 | ||
|   | cc1c5e45b2 | ||
|   | 038199c447 | ||
|   | 8a1eab7ceb | ||
|   | bc5bd35448 | ||
|   | 1795fd56b7 | ||
|   | 3d788d6056 | ||
|   | 4a7c33edad | ||
|   | 797f60d725 | ||
|   | 2427d68aa1 | ||
|   | 00c6b0f8ed | ||
|   | 7560f98d70 | ||
|   | 7b8d4ab3d6 | ||
|   | 07a1a805f6 | ||
|   | d8bab6aba9 | ||
|   | a930e2dc75 | ||
|   | 2eb35668fa | ||
|   | 07f4e5ac5c | ||
|   | 1533c22d5c | ||
|   | db82a90414 | ||
|   | 51a693badf | ||
|   | 2aa8f5b352 | ||
|   | 93b3b8f985 | ||
|   | 92c8bd80b5 | ||
|   | 528af0157d | ||
|   | 10a77b6278 | ||
|   | 03bbf6a582 | ||
|   | 63fcb649d2 | ||
|   | 4f60a92b92 | ||
|   | 0419c1a41f | ||
|   | 2d5ae78521 | ||
|   | 959134df02 | ||
|   | cf03f103ab | ||
|   | a9f9fc4ce2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cfb370a3c8 | ||
|   | 353435c8d5 | ||
|   | c8c85d096b | ||
|   | 19c9c8f227 | ||
|   | 6ea2a29eea | ||
|   | 59f3f819a6 | ||
|   | 93e8f52880 | ||
|   | 02810efcc4 | ||
|   | 4a8a7c997e | ||
|   | 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 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 29e8d1cff0 | ||
|   | 4e1d10cc08 | ||
|   | 3575d94ca1 | ||
|   | d91546b532 | ||
|   | 9f554f4917 | ||
|   | d4720a9244 | ||
|   | 5c466712db | ||
|   | 6dc7e852ae | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 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 | ||
|   | 9612bc78fe | ||
|   | d408e8653c | ||
|   | cc76ccc3c9 | ||
|   | 105a00d3e4 | ||
|   | 2c08cba8cc | ||
|   | 344b11a204 | ||
|   | 1ff5bf0fd5 | ||
|   | 2b86137388 | ||
|   | c29cf7f77c | ||
|   | 193cb46d60 | ||
|   | 9dc864d486 | ||
|   | cee166839a | ||
|   | 1a60a3c728 | ||
|   | 5d946778cb | ||
|   | ac5f85820f | ||
|   | 716e100a28 | ||
|   | 7b8cb16c12 | ||
|   | 00d46424a3 | ||
|   | 2a5f940744 | ||
|   | 13cc016b36 | ||
|   | a8d49c27c8 | ||
|   | 8fdbe447c1 | ||
|   | a8522e91b5 | ||
|   | 5754f4463d | ||
|   | d4118ade0f | ||
|   | 6d80f15a98 | ||
|   | f8aa472409 | ||
|   | df22fd00ca | ||
|   | ce2743a982 | ||
|   | 92b32458ad | ||
|   | d57e8a45d3 | ||
|   | 551d3ffdf3 | ||
|   | 7add6eb736 | ||
|   | a28616d535 | ||
|   | a288fd370f | ||
|   | acd335e249 | ||
|   | 764ae7e0b6 | ||
|   | da0bfa1945 | ||
|   | 3c61d709b5 | ||
|   | ffc92a7b63 | ||
|   | af0c7b5a50 | ||
|   | 1904c4d057 | ||
|   | 542f169b36 | ||
|   | 65a30bf60c | ||
|   | 2e51da32f0 | ||
|   | 0562242043 | ||
|   | debcdefc21 | ||
|   | 6b7e78320d | ||
|   | 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 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 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 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 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 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 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 | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										124
									
								
								.eslintrc.json
									
									
									
									
									
								
							
							
						
						
									
										124
									
								
								.eslintrc.json
									
									
									
									
									
								
							| @@ -1,11 +1,11 @@ | ||||
| { | ||||
|   "extends": [ | ||||
|     "airbnb-base", | ||||
|     "airbnb-typescript/base", | ||||
|     "plugin:@typescript-eslint/recommended", | ||||
|     "plugin:wc/recommended", | ||||
|     "plugin:lit/recommended", | ||||
|     "prettier", | ||||
|     "prettier/@typescript-eslint" | ||||
|     "plugin:lit/all", | ||||
|     "prettier" | ||||
|   ], | ||||
|   "parser": "@typescript-eslint/parser", | ||||
|   "parserOptions": { | ||||
| @@ -29,63 +29,91 @@ | ||||
|     "__BUILD__": false, | ||||
|     "__VERSION__": false, | ||||
|     "__STATIC_PATH__": false, | ||||
|     "Polymer": true, | ||||
|     "webkitSpeechRecognition": false, | ||||
|     "ResizeObserver": false | ||||
|     "Polymer": true | ||||
|   }, | ||||
|   "env": { | ||||
|     "browser": true, | ||||
|     "es6": true | ||||
|   }, | ||||
|   "rules": { | ||||
|     "class-methods-use-this": 0, | ||||
|     "new-cap": 0, | ||||
|     "prefer-template": 0, | ||||
|     "object-shorthand": 0, | ||||
|     "func-names": 0, | ||||
|     "prefer-arrow-callback": 0, | ||||
|     "no-underscore-dangle": 0, | ||||
|     "strict": 0, | ||||
|     "prefer-spread": 0, | ||||
|     "no-plusplus": 0, | ||||
|     "no-bitwise": 2, | ||||
|     "comma-dangle": 0, | ||||
|     "vars-on-top": 0, | ||||
|     "no-continue": 0, | ||||
|     "no-param-reassign": 0, | ||||
|     "no-multi-assign": 0, | ||||
|     "no-console": 2, | ||||
|     "radix": 0, | ||||
|     "no-alert": 0, | ||||
|     "no-return-await": 0, | ||||
|     "no-nested-ternary": 0, | ||||
|     "prefer-destructuring": 0, | ||||
|     "class-methods-use-this": "off", | ||||
|     "new-cap": "off", | ||||
|     "prefer-template": "off", | ||||
|     "object-shorthand": "off", | ||||
|     "func-names": "off", | ||||
|     "no-underscore-dangle": "off", | ||||
|     "strict": "off", | ||||
|     "no-plusplus": "off", | ||||
|     "no-bitwise": "error", | ||||
|     "comma-dangle": "off", | ||||
|     "vars-on-top": "off", | ||||
|     "no-continue": "off", | ||||
|     "no-param-reassign": "off", | ||||
|     "no-multi-assign": "off", | ||||
|     "no-console": "error", | ||||
|     "radix": "off", | ||||
|     "no-alert": "off", | ||||
|     "no-nested-ternary": "off", | ||||
|     "prefer-destructuring": "off", | ||||
|     "no-restricted-globals": [2, "event"], | ||||
|     "prefer-promise-reject-errors": 0, | ||||
|     "import/order": 0, | ||||
|     "import/prefer-default-export": 0, | ||||
|     "import/no-unresolved": 0, | ||||
|     "import/no-cycle": 0, | ||||
|     "prefer-promise-reject-errors": "off", | ||||
|     "import/prefer-default-export": "off", | ||||
|     "import/no-default-export": "off", | ||||
|     "import/no-unresolved": "off", | ||||
|     "import/no-cycle": "off", | ||||
|     "import/extensions": [ | ||||
|       2, | ||||
|       "error", | ||||
|       "ignorePackages", | ||||
|       { "ts": "never", "js": "never" } | ||||
|     ], | ||||
|     "no-restricted-syntax": ["error", "LabeledStatement", "WithStatement"], | ||||
|     "object-curly-newline": 0, | ||||
|     "default-case": 0, | ||||
|     "wc/no-self-class": 0, | ||||
|     "no-shadow": 0, | ||||
|     "@typescript-eslint/camelcase": 0, | ||||
|     "@typescript-eslint/ban-ts-comment": 0, | ||||
|     "@typescript-eslint/no-use-before-define": 0, | ||||
|     "@typescript-eslint/no-non-null-assertion": 0, | ||||
|     "@typescript-eslint/no-explicit-any": 0, | ||||
|     "@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"] | ||||
|     "object-curly-newline": "off", | ||||
|     "default-case": "off", | ||||
|     "wc/no-self-class": "off", | ||||
|     "no-shadow": "off", | ||||
|     "@typescript-eslint/camelcase": "off", | ||||
|     "@typescript-eslint/ban-ts-comment": "off", | ||||
|     "@typescript-eslint/no-use-before-define": "off", | ||||
|     "@typescript-eslint/no-non-null-assertion": "off", | ||||
|     "@typescript-eslint/no-explicit-any": "off", | ||||
|     "@typescript-eslint/explicit-function-return-type": "off", | ||||
|     "@typescript-eslint/explicit-module-boundary-types": "off", | ||||
|     "@typescript-eslint/no-shadow": ["error"], | ||||
|     "@typescript-eslint/naming-convention": [ | ||||
|       "off", | ||||
|       { | ||||
|         "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"] | ||||
|       } | ||||
|     ], | ||||
|     "@typescript-eslint/no-unused-vars": "off", | ||||
|     "unused-imports/no-unused-vars": [ | ||||
|       "error", | ||||
|       { | ||||
|         "vars": "all", | ||||
|         "varsIgnorePattern": "^_", | ||||
|         "args": "after-used", | ||||
|         "argsIgnorePattern": "^_", | ||||
|         "ignoreRestSiblings": true | ||||
|       } | ||||
|     ], | ||||
|     "unused-imports/no-unused-imports": "error", | ||||
|     "lit/attribute-value-entities": "off", | ||||
|     "lit/no-template-map": "off", | ||||
|     "lit/no-template-arrow": "warn" | ||||
|   }, | ||||
|   "plugins": ["disable", "import", "lit", "prettier", "@typescript-eslint"], | ||||
|   "plugins": ["disable", "unused-imports"], | ||||
|   "processor": "disable/disable" | ||||
| } | ||||
|   | ||||
| @@ -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. | ||||
							
								
								
									
										84
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										84
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -10,26 +10,21 @@ on: | ||||
|       - dev | ||||
|       - master | ||||
|  | ||||
| env: | ||||
|   NODE_VERSION: 14 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| jobs: | ||||
|   lint: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Setting up Node.js | ||||
|         uses: actions/setup-node@v1 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 12.x | ||||
|       - name: Get yarn cache path | ||||
|         id: yarn-cache-dir-path | ||||
|         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||
|       - name: Fetching Yarn cache | ||||
|         uses: actions/cache@v1 | ||||
|         with: | ||||
|           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||
|           restore-keys: | | ||||
|             ${{ runner.os }}-yarn- | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install | ||||
|         env: | ||||
| @@ -37,54 +32,40 @@ 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 | ||||
|       - name: Check for duplicate dependencies | ||||
|         run: yarn dedupe --check | ||||
|   test: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Setting up Node.js | ||||
|         uses: actions/setup-node@v1 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 12.x | ||||
|       - name: Get yarn cache path | ||||
|         id: yarn-cache-dir-path | ||||
|         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||
|       - name: Fetching Yarn cache | ||||
|         uses: actions/cache@v1 | ||||
|         with: | ||||
|           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||
|           restore-keys: | | ||||
|             ${{ runner.os }}-yarn- | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install | ||||
|         env: | ||||
|           CI: true | ||||
|       - name: Run Mocha | ||||
|         run: npm run mocha | ||||
|       - name: Run Tests | ||||
|         run: yarn run test | ||||
|   build: | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: [lint, test] | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Setting up Node.js | ||||
|         uses: actions/setup-node@v1 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 12.x | ||||
|       - name: Get yarn cache path | ||||
|         id: yarn-cache-dir-path | ||||
|         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||
|       - name: Fetching Yarn cache | ||||
|         uses: actions/cache@v1 | ||||
|         with: | ||||
|           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||
|           restore-keys: | | ||||
|             ${{ runner.os }}-yarn- | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install | ||||
|         env: | ||||
| @@ -99,20 +80,11 @@ jobs: | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Setting up Node.js | ||||
|         uses: actions/setup-node@v1 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 12.x | ||||
|       - name: Get yarn cache path | ||||
|         id: yarn-cache-dir-path | ||||
|         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||
|       - name: Fetching Yarn cache | ||||
|         uses: actions/cache@v1 | ||||
|         with: | ||||
|           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||
|           restore-keys: | | ||||
|             ${{ runner.os }}-yarn- | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install | ||||
|         env: | ||||
|   | ||||
							
								
								
									
										22
									
								
								.github/workflows/demo.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/demo.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -4,26 +4,22 @@ on: | ||||
|   push: | ||||
|     branches: | ||||
|       - dev | ||||
|  | ||||
| env: | ||||
|   NODE_VERSION: 14 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| jobs: | ||||
|   deploy: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Check out files from GitHub | ||||
|         uses: actions/checkout@v2 | ||||
|       - name: Setting up Node.js | ||||
|         uses: actions/setup-node@v1 | ||||
|       - name: Set up Node ${{ env.NODE_VERSION }} | ||||
|         uses: actions/setup-node@v2 | ||||
|         with: | ||||
|           node-version: 12.x | ||||
|       - name: Get yarn cache path | ||||
|         id: yarn-cache-dir-path | ||||
|         run: echo "::set-output name=dir::$(yarn cache dir)" | ||||
|       - name: Fetching Yarn cache | ||||
|         uses: actions/cache@v1 | ||||
|         with: | ||||
|           path: ${{ steps.yarn-cache-dir-path.outputs.dir }} | ||||
|           key: ${{ runner.os }}-yarn-${{ hashFiles('**/yarn.lock') }} | ||||
|           restore-keys: | | ||||
|             ${{ runner.os }}-yarn- | ||||
|           node-version: ${{ env.NODE_VERSION }} | ||||
|           cache: yarn | ||||
|       - name: Install dependencies | ||||
|         run: yarn install | ||||
|         env: | ||||
|   | ||||
							
								
								
									
										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 }} | ||||
							
								
								
									
										91
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| name: Release | ||||
|  | ||||
| on: | ||||
|   release: | ||||
|     types: | ||||
|       - published | ||||
|  | ||||
| env: | ||||
|   PYTHON_VERSION: 3.8 | ||||
|   NODE_VERSION: 14 | ||||
|   NODE_OPTIONS: --max_old_space_size=6144 | ||||
|  | ||||
| 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 }} | ||||
|           cache: yarn | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: yarn install | ||||
|  | ||||
|       - name: Download Translations | ||||
|         run: ./script/translations_download | ||||
|         env: | ||||
|           LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }} | ||||
|       - 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.9-alpine3.14" | ||||
|     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" | ||||
							
								
								
									
										25
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| name: Translations | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - dev | ||||
|     paths: | ||||
|       - src/translations/en.json | ||||
|  | ||||
| env: | ||||
|   NODE_VERSION: 14 | ||||
|  | ||||
| jobs: | ||||
|   upload: | ||||
|     name: Upload | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout the repository | ||||
|         uses: actions/checkout@v2 | ||||
|  | ||||
|       - name: Upload Translations | ||||
|         run: | | ||||
|           export LOKALISE_TOKEN="${{ secrets.LOKALISE_TOKEN }}" | ||||
|  | ||||
|           ./script/translations_upload_base | ||||
							
								
								
									
										29
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,10 +1,23 @@ | ||||
| .DS_Store | ||||
| .reify-cache | ||||
|  | ||||
| # build | ||||
| build | ||||
| build-translations/* | ||||
| node_modules/* | ||||
| npm-debug.log | ||||
| .DS_Store | ||||
| hass_frontend/* | ||||
| .reify-cache | ||||
| dist | ||||
|  | ||||
| # yarn | ||||
| .yarn/* | ||||
| !.yarn/patches | ||||
| !.yarn/releases | ||||
| !.yarn/plugins | ||||
| !.yarn/sdks | ||||
| !.yarn/versions | ||||
| .pnp.* | ||||
| node_modules/* | ||||
| yarn-error.log | ||||
| npm-debug.log | ||||
|  | ||||
| # Python stuff | ||||
| *.py[cod] | ||||
| @@ -14,11 +27,8 @@ hass_frontend/* | ||||
| # venv stuff | ||||
| pyvenv.cfg | ||||
| pip-selfcheck.json | ||||
| venv | ||||
| venv/* | ||||
| .venv | ||||
| lib | ||||
| bin | ||||
| dist | ||||
|  | ||||
| # vscode | ||||
| .vscode/* | ||||
| @@ -31,9 +41,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 | ||||
							
								
								
									
										1536
									
								
								.yarn/patches/@lit-labs/virtualizer/0.7.0.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1536
									
								
								.yarn/patches/@lit-labs/virtualizer/0.7.0.patch
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										29
									
								
								.yarn/patches/@lit-labs/virtualizer/event-target-shim.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								.yarn/patches/@lit-labs/virtualizer/event-target-shim.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| diff --git a/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js b/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js | ||||
| index d92179f7fd5315203f870a6963e871dc8ddf6c0c..362e284121b97e0fba0925225777aebc32e26b8d 100644 | ||||
| --- a/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js | ||||
| +++ b/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js | ||||
| @@ -1,14 +1,15 @@ | ||||
| -let _ET, ET; | ||||
| +let _ET; | ||||
| +let ET; | ||||
|  export default async function EventTarget() { | ||||
| -    return ET || init(); | ||||
| +  return ET || init(); | ||||
|  } | ||||
|  async function init() { | ||||
| -    _ET = window.EventTarget; | ||||
| -    try { | ||||
| -        new _ET(); | ||||
| -    } | ||||
| -    catch (_a) { | ||||
| -        _ET = (await import('event-target-shim')).EventTarget; | ||||
| -    } | ||||
| -    return (ET = _ET); | ||||
| +  _ET = window.EventTarget; | ||||
| +  try { | ||||
| +    new _ET(); | ||||
| +  } catch (_a) { | ||||
| +    _ET = (await import("event-target-shim")).default.EventTarget; | ||||
| +  } | ||||
| +  return (ET = _ET); | ||||
|  } | ||||
							
								
								
									
										34
									
								
								.yarn/patches/@polymer/polymer/pr-5569.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								.yarn/patches/@polymer/polymer/pr-5569.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| diff --git a/lib/legacy/class.js b/lib/legacy/class.js | ||||
| index aee2511be1cd9bf900ee552bc98190c1631c57c0..f2f499d68bf52034cac9c28307c99e8ce6b8417d 100644 | ||||
| --- a/lib/legacy/class.js | ||||
| +++ b/lib/legacy/class.js | ||||
| @@ -304,17 +304,23 @@ function GenerateClassFromInfo(info, Base, behaviors) { | ||||
|        // only proceed if the generated class' prototype has not been registered. | ||||
|        const generatedProto = PolymerGenerated.prototype; | ||||
|        if (!generatedProto.hasOwnProperty(JSCompiler_renameProperty('__hasRegisterFinished', generatedProto))) { | ||||
| -        generatedProto.__hasRegisterFinished = true; | ||||
| +        // make sure legacy lifecycle is called on the *element*'s prototype | ||||
| +        // and not the generated class prototype; if the element has been | ||||
| +        // extended, these are *not* the same. | ||||
| +        const proto = Object.getPrototypeOf(this); | ||||
| +        // Only set flag when generated prototype itself is registered, | ||||
| +        // as this element may be extended from, and needs to run `registered` | ||||
| +        // on all behaviors on the subclass as well. | ||||
| +        if (proto === generatedProto) { | ||||
| +          generatedProto.__hasRegisterFinished = true; | ||||
| +        } | ||||
|          // ensure superclass is registered first. | ||||
|          super._registered(); | ||||
|          // copy properties onto the generated class lazily if we're optimizing, | ||||
| -        if (legacyOptimizations) { | ||||
| +        if (legacyOptimizations && !Object.hasOwnProperty(generatedProto, '__hasCopiedProperties')) { | ||||
| +          generatedProto.__hasCopiedProperties = true; | ||||
|            copyPropertiesToProto(generatedProto); | ||||
|          } | ||||
| -        // make sure legacy lifecycle is called on the *element*'s prototype | ||||
| -        // and not the generated class prototype; if the element has been | ||||
| -        // extended, these are *not* the same. | ||||
| -        const proto = Object.getPrototypeOf(this); | ||||
|          let list = lifecycle.beforeRegister; | ||||
|          if (list) { | ||||
|            for (let i=0; i < list.length; i++) { | ||||
							
								
								
									
										77
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										8
									
								
								.yarn/plugins/@yarnpkg/plugin-typescript.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.yarn/plugins/@yarnpkg/plugin-typescript.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										631
									
								
								.yarn/releases/yarn-3.0.2.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										631
									
								
								.yarn/releases/yarn-3.0.2.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										9
									
								
								.yarnrc.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.yarnrc.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| nodeLinker: node-modules | ||||
|  | ||||
| plugins: | ||||
|   - path: .yarn/plugins/@yarnpkg/plugin-typescript.cjs | ||||
|     spec: "@yarnpkg/plugin-typescript" | ||||
|   - path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs | ||||
|     spec: "@yarnpkg/plugin-interactive-tools" | ||||
|  | ||||
| yarnPath: .yarn/releases/yarn-3.0.2.cjs | ||||
| @@ -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). | ||||
							
								
								
									
										170
									
								
								build-scripts/babel-plugins/inline-constants-plugin.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										170
									
								
								build-scripts/babel-plugins/inline-constants-plugin.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,170 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const path = require("path"); | ||||
|  | ||||
| // Currently only supports CommonJS modules, as require is synchronous. `import` would need babel running asynchronous. | ||||
| module.exports = function inlineConstants(babel, options, cwd) { | ||||
|   const t = babel.types; | ||||
|  | ||||
|   if (!Array.isArray(options.modules)) { | ||||
|     throw new TypeError( | ||||
|       "babel-plugin-inline-constants: expected a `modules` array to be passed" | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   if (options.resolveExtensions && !Array.isArray(options.resolveExtensions)) { | ||||
|     throw new TypeError( | ||||
|       "babel-plugin-inline-constants: expected `resolveExtensions` to be an array" | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   const ignoreModuleNotFound = options.ignoreModuleNotFound; | ||||
|   const resolveExtensions = options.resolveExtensions; | ||||
|  | ||||
|   const hasRelativeModules = options.modules.some( | ||||
|     (module) => module.startsWith(".") || module.startsWith("/") | ||||
|   ); | ||||
|  | ||||
|   const modules = Object.fromEntries( | ||||
|     options.modules.map((module) => { | ||||
|       const absolute = module.startsWith(".") | ||||
|         ? require.resolve(module, { paths: [cwd] }) | ||||
|         : module; | ||||
|       // eslint-disable-next-line import/no-dynamic-require | ||||
|       return [absolute, require(absolute)]; | ||||
|     }) | ||||
|   ); | ||||
|  | ||||
|   const toLiteral = (value) => { | ||||
|     if (typeof value === "string") { | ||||
|       return t.stringLiteral(value); | ||||
|     } | ||||
|  | ||||
|     if (typeof value === "number") { | ||||
|       return t.numericLiteral(value); | ||||
|     } | ||||
|  | ||||
|     if (typeof value === "boolean") { | ||||
|       return t.booleanLiteral(value); | ||||
|     } | ||||
|  | ||||
|     if (value === null) { | ||||
|       return t.nullLiteral(); | ||||
|     } | ||||
|  | ||||
|     throw new Error( | ||||
|       "babel-plugin-inline-constants: cannot handle non-literal `" + value + "`" | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
|   const resolveAbsolute = (value, state, resolveExtensionIndex) => { | ||||
|     if (!state.filename) { | ||||
|       throw new TypeError( | ||||
|         "babel-plugin-inline-constants: expected a `filename` to be set for files" | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     if (resolveExtensions && resolveExtensionIndex !== undefined) { | ||||
|       value += resolveExtensions[resolveExtensionIndex]; | ||||
|     } | ||||
|  | ||||
|     try { | ||||
|       return require.resolve(value, { paths: [path.dirname(state.filename)] }); | ||||
|     } catch (error) { | ||||
|       if ( | ||||
|         error.code === "MODULE_NOT_FOUND" && | ||||
|         resolveExtensions && | ||||
|         (resolveExtensionIndex === undefined || | ||||
|           resolveExtensionIndex < resolveExtensions.length - 1) | ||||
|       ) { | ||||
|         const resolveExtensionIdx = (resolveExtensionIndex || -1) + 1; | ||||
|         return resolveAbsolute(value, state, resolveExtensionIdx); | ||||
|       } | ||||
|  | ||||
|       if (error.code === "MODULE_NOT_FOUND" && ignoreModuleNotFound) { | ||||
|         return undefined; | ||||
|       } | ||||
|       throw error; | ||||
|     } | ||||
|   }; | ||||
|  | ||||
|   const importDeclaration = (p, state) => { | ||||
|     if (p.node.type !== "ImportDeclaration") { | ||||
|       return; | ||||
|     } | ||||
|     const absolute = | ||||
|       hasRelativeModules && p.node.source.value.startsWith(".") | ||||
|         ? resolveAbsolute(p.node.source.value, state) | ||||
|         : p.node.source.value; | ||||
|  | ||||
|     if (!absolute || !(absolute in modules)) { | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     const module = modules[absolute]; | ||||
|  | ||||
|     for (const specifier of p.node.specifiers) { | ||||
|       if ( | ||||
|         specifier.type === "ImportDefaultSpecifier" && | ||||
|         specifier.local && | ||||
|         specifier.local.type === "Identifier" | ||||
|       ) { | ||||
|         if (!("default" in module)) { | ||||
|           throw new Error( | ||||
|             "babel-plugin-inline-constants: cannot access default export from `" + | ||||
|               p.node.source.value + | ||||
|               "`" | ||||
|           ); | ||||
|         } | ||||
|  | ||||
|         const variableValue = toLiteral(module.default); | ||||
|         const variable = t.variableDeclarator( | ||||
|           t.identifier(specifier.local.name), | ||||
|           variableValue | ||||
|         ); | ||||
|  | ||||
|         p.insertBefore({ | ||||
|           type: "VariableDeclaration", | ||||
|           kind: "const", | ||||
|           declarations: [variable], | ||||
|         }); | ||||
|       } else if ( | ||||
|         specifier.type === "ImportSpecifier" && | ||||
|         specifier.imported && | ||||
|         specifier.imported.type === "Identifier" && | ||||
|         specifier.local && | ||||
|         specifier.local.type === "Identifier" | ||||
|       ) { | ||||
|         if (!(specifier.imported.name in module)) { | ||||
|           throw new Error( | ||||
|             "babel-plugin-inline-constants: cannot access `" + | ||||
|               specifier.imported.name + | ||||
|               "` from `" + | ||||
|               p.node.source.value + | ||||
|               "`" | ||||
|           ); | ||||
|         } | ||||
|  | ||||
|         const variableValue = toLiteral(module[specifier.imported.name]); | ||||
|         const variable = t.variableDeclarator( | ||||
|           t.identifier(specifier.local.name), | ||||
|           variableValue | ||||
|         ); | ||||
|  | ||||
|         p.insertBefore({ | ||||
|           type: "VariableDeclaration", | ||||
|           kind: "const", | ||||
|           declarations: [variable], | ||||
|         }); | ||||
|       } else { | ||||
|         throw new Error("Cannot handle specifier `" + specifier.type + "`"); | ||||
|       } | ||||
|     } | ||||
|     p.remove(); | ||||
|   }; | ||||
|  | ||||
|   return { | ||||
|     visitor: { | ||||
|       ImportDeclaration: importDeclaration, | ||||
|     }, | ||||
|   }; | ||||
| }; | ||||
| @@ -1,11 +1,10 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| const path = require("path"); | ||||
| const env = require("./env.js"); | ||||
| const paths = require("./paths.js"); | ||||
|  | ||||
| // Files from NPM Packages that should not be imported | ||||
| module.exports.ignorePackages = ({ latestBuild }) => [ | ||||
|   // Bloats bundle and it's not used. | ||||
|   path.resolve(require.resolve("moment"), "../locale"), | ||||
|   // Part of yaml.js and only used for !!js functions that we don't use | ||||
|   require.resolve("esprima"), | ||||
| ]; | ||||
| @@ -19,7 +18,8 @@ module.exports.emptyPackages = ({ latestBuild }) => | ||||
|     require.resolve("@polymer/paper-styles/default-theme.js"), | ||||
|     // Loads stuff from a CDN | ||||
|     require.resolve("@polymer/font-roboto/roboto.js"), | ||||
|     require.resolve("@vaadin/vaadin-material-styles/font-roboto.js"), | ||||
|     require.resolve("@vaadin/vaadin-material-styles/typography.js"), | ||||
|     require.resolve("@vaadin/vaadin-material-styles/font-icons.js"), | ||||
|     // Compatibility not needed for latest builds | ||||
|     latestBuild && | ||||
|       // wrapped in require.resolve so it blows up if file no longer exists | ||||
| @@ -44,24 +44,36 @@ 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", | ||||
|         corejs: "3.15", | ||||
|         bugfixes: true, | ||||
|       }, | ||||
|     ], | ||||
|     require("@babel/preset-typescript").default, | ||||
|     "@babel/preset-typescript", | ||||
|   ].filter(Boolean), | ||||
|   plugins: [ | ||||
|     [ | ||||
|       path.resolve( | ||||
|         paths.polymer_dir, | ||||
|         "build-scripts/babel-plugins/inline-constants-plugin.js" | ||||
|       ), | ||||
|       { | ||||
|         modules: ["@mdi/js"], | ||||
|         ignoreModuleNotFound: true, | ||||
|       }, | ||||
|     ], | ||||
|     // Part of ES2018. Converts {...a, b: 2} to Object.assign({}, a, {b: 2}) | ||||
|     !latestBuild && [ | ||||
|       "@babel/plugin-proposal-object-rest-spread", | ||||
| @@ -70,25 +82,21 @@ module.exports.babelOptions = ({ latestBuild }) => ({ | ||||
|     // Only support the syntax, Webpack will handle it. | ||||
|     "@babel/plugin-syntax-import-meta", | ||||
|     "@babel/plugin-syntax-dynamic-import", | ||||
|     "@babel/plugin-syntax-top-level-await", | ||||
|     "@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-private-property-in-object", { loose: true }], | ||||
|     ["@babel/plugin-proposal-class-properties", { loose: true }], | ||||
|   ].filter(Boolean), | ||||
|   exclude: [ | ||||
|     // \\ for Windows, / for Mac OS and Linux | ||||
|     /node_modules[\\/]core-js/, | ||||
|     /node_modules[\\/]webpack[\\/]buildin/, | ||||
|   ], | ||||
| }); | ||||
|  | ||||
| // 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 +125,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 +140,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" | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| @@ -42,8 +47,8 @@ gulp.task( | ||||
|     gulp.parallel("gen-icons-json", "build-translations"), | ||||
|     "copy-static-app", | ||||
|     env.useRollup() ? "rollup-prod-app" : "webpack-prod-app", | ||||
|     ...// Don't compress running tests | ||||
|     (env.isTest() ? [] : ["compress-app"]), | ||||
|     // Don't compress running tests | ||||
|     ...(env.isTest() ? [] : ["compress-app"]), | ||||
|     gulp.parallel( | ||||
|       "gen-pages-prod", | ||||
|       "gen-index-app-prod", | ||||
|   | ||||
| @@ -5,32 +5,32 @@ require("./translations"); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean", | ||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||
|     return del([paths.app_output_root, paths.build_dir]); | ||||
|   }) | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.app_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-demo", | ||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||
|     return del([paths.demo_output_root, paths.build_dir]); | ||||
|   }) | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.demo_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-cast", | ||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||
|     return del([paths.cast_output_root, paths.build_dir]); | ||||
|   }) | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.cast_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task("clean-hassio", function cleanOutputAndBuildDir() { | ||||
|   return del([paths.hassio_output_root, paths.build_dir]); | ||||
| }); | ||||
| gulp.task("clean-hassio", () => | ||||
|   del([paths.hassio_output_root, paths.build_dir]) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
|   "clean-gallery", | ||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||
|     return del([paths.gallery_output_root, paths.build_dir]); | ||||
|   }) | ||||
|   gulp.parallel("clean-translations", () => | ||||
|     del([paths.gallery_output_root, paths.build_dir]) | ||||
|   ) | ||||
| ); | ||||
|   | ||||
| @@ -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", | ||||
| @@ -288,15 +302,23 @@ gulp.task("gen-index-hassio-prod", async () => { | ||||
|  | ||||
| function writeHassioEntrypoint(latestEntrypoint, es5Entrypoint) { | ||||
|   fs.mkdirSync(paths.hassio_output_root, { recursive: true }); | ||||
|   // Safari 12 and below does not have a compliant ES2015 implementation of template literals, so we ship ES5 | ||||
|   fs.writeFileSync( | ||||
|     path.resolve(paths.hassio_output_root, "entrypoint.js"), | ||||
|     ` | ||||
| try { | ||||
|   new Function("import('${latestEntrypoint}')")(); | ||||
| } catch (err) { | ||||
| function loadES5() { | ||||
|   var el = document.createElement('script'); | ||||
|   el.src = '${es5Entrypoint}'; | ||||
|   document.body.appendChild(el); | ||||
| } | ||||
| if (/.*Version\\/(?:11|12)(?:\\.\\d+)*.*Safari\\//.test(navigator.userAgent)) { | ||||
|     loadES5(); | ||||
| } else { | ||||
|   try { | ||||
|     new Function("import('${latestEntrypoint}')")(); | ||||
|   } catch (err) { | ||||
|     loadES5(); | ||||
|   } | ||||
| } | ||||
|   `, | ||||
|     { encoding: "utf-8" } | ||||
|   | ||||
| @@ -2,7 +2,6 @@ | ||||
|  | ||||
| const gulp = require("gulp"); | ||||
| const path = require("path"); | ||||
| const cpx = require("cpx"); | ||||
| const fs = require("fs-extra"); | ||||
| const paths = require("../paths"); | ||||
|  | ||||
| @@ -13,8 +12,10 @@ const polyPath = (...parts) => path.resolve(paths.polymer_dir, ...parts); | ||||
| const copyFileDir = (fromFile, toDir) => | ||||
|   fs.copySync(fromFile, path.join(toDir, path.basename(fromFile))); | ||||
|  | ||||
| const genStaticPath = (staticDir) => (...parts) => | ||||
|   path.resolve(staticDir, ...parts); | ||||
| const genStaticPath = | ||||
|   (staticDir) => | ||||
|   (...parts) => | ||||
|     path.resolve(staticDir, ...parts); | ||||
|  | ||||
| function copyTranslations(staticDir) { | ||||
|   const staticPath = genStaticPath(staticDir); | ||||
| @@ -62,9 +63,12 @@ function copyLoaderJS(staticDir) { | ||||
| function copyFonts(staticDir) { | ||||
|   const staticPath = genStaticPath(staticDir); | ||||
|   // Local fonts | ||||
|   cpx.copySync( | ||||
|     npmPath("roboto-fontface/fonts/roboto/*.woff2"), | ||||
|     staticPath("fonts/roboto") | ||||
|   fs.copySync( | ||||
|     npmPath("roboto-fontface/fonts/roboto/"), | ||||
|     staticPath("fonts/roboto/"), | ||||
|     { | ||||
|       filter: (src) => !src.includes(".") || src.endsWith(".woff2"), | ||||
|     } | ||||
|   ); | ||||
| } | ||||
|  | ||||
| @@ -85,6 +89,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 | ||||
|   | ||||
| @@ -1,3 +1,5 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
|  | ||||
| const crypto = require("crypto"); | ||||
| const del = require("del"); | ||||
| const path = require("path"); | ||||
| @@ -26,32 +28,14 @@ gulp.task("translations-enable-merge-backend", (done) => { | ||||
|   done(); | ||||
| }); | ||||
|  | ||||
| String.prototype.rsplit = function (sep, maxsplit) { | ||||
|   var split = this.split(sep); | ||||
|   return maxsplit | ||||
|     ? [split.slice(0, -maxsplit).join(sep)].concat(split.slice(-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 = {}; | ||||
|   Object.keys(data).forEach(function (key) { | ||||
|   Object.keys(data).forEach((key) => { | ||||
|     if (typeof data[key] === "object") { | ||||
|       output = { | ||||
|         ...output, | ||||
| @@ -112,15 +96,19 @@ function lokaliseTransform(data, original, file) { | ||||
|     if (value instanceof Object) { | ||||
|       output[key] = lokaliseTransform(value, original, file); | ||||
|     } else { | ||||
|       output[key] = value.replace(re_key_reference, (match, key) => { | ||||
|         const replace = key.split("::").reduce((tr, k) => { | ||||
|       output[key] = value.replace(re_key_reference, (_match, lokalise_key) => { | ||||
|         const replace = lokalise_key.split("::").reduce((tr, k) => { | ||||
|           if (!tr) { | ||||
|             throw Error(`Invalid key placeholder ${key} in ${file.path}`); | ||||
|             throw Error( | ||||
|               `Invalid key placeholder ${lokalise_key} in ${file.path}` | ||||
|             ); | ||||
|           } | ||||
|           return tr[k]; | ||||
|         }, original); | ||||
|         if (typeof replace !== "string") { | ||||
|           throw Error(`Invalid key placeholder ${key} in ${file.path}`); | ||||
|           throw Error( | ||||
|             `Invalid key placeholder ${lokalise_key} in ${file.path}` | ||||
|           ); | ||||
|         } | ||||
|         return replace; | ||||
|       }); | ||||
| @@ -129,9 +117,7 @@ function lokaliseTransform(data, original, file) { | ||||
|   return output; | ||||
| } | ||||
|  | ||||
| gulp.task("clean-translations", function () { | ||||
|   return del([workDir]); | ||||
| }); | ||||
| gulp.task("clean-translations", () => del([workDir])); | ||||
|  | ||||
| gulp.task("ensure-translations-build-dir", (done) => { | ||||
|   if (!fs.existsSync(workDir)) { | ||||
| @@ -140,7 +126,7 @@ gulp.task("ensure-translations-build-dir", (done) => { | ||||
|   done(); | ||||
| }); | ||||
|  | ||||
| gulp.task("create-test-metadata", function (cb) { | ||||
| gulp.task("create-test-metadata", (cb) => { | ||||
|   fs.writeFile( | ||||
|     workDir + "/testMetadata.json", | ||||
|     JSON.stringify({ | ||||
| @@ -154,17 +140,13 @@ gulp.task("create-test-metadata", function (cb) { | ||||
|  | ||||
| gulp.task( | ||||
|   "create-test-translation", | ||||
|   gulp.series("create-test-metadata", function createTestTranslation() { | ||||
|     return gulp | ||||
|   gulp.series("create-test-metadata", () => | ||||
|     gulp | ||||
|       .src(path.join(paths.translations_src, "en.json")) | ||||
|       .pipe( | ||||
|         transform(function (data, file) { | ||||
|           return recursiveEmpty(data); | ||||
|         }) | ||||
|       ) | ||||
|       .pipe(transform((data, _file) => recursiveEmpty(data))) | ||||
|       .pipe(rename("test.json")) | ||||
|       .pipe(gulp.dest(workDir)); | ||||
|   }) | ||||
|       .pipe(gulp.dest(workDir)) | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| /** | ||||
| @@ -176,7 +158,7 @@ gulp.task( | ||||
|  * project is buildable immediately after merging new translation keys, since | ||||
|  * the Lokalise update to translations/en.json will not happen immediately. | ||||
|  */ | ||||
| gulp.task("build-master-translation", function () { | ||||
| gulp.task("build-master-translation", () => { | ||||
|   const src = [path.join(paths.translations_src, "en.json")]; | ||||
|  | ||||
|   if (mergeBackend) { | ||||
| @@ -185,11 +167,7 @@ gulp.task("build-master-translation", function () { | ||||
|  | ||||
|   return gulp | ||||
|     .src(src) | ||||
|     .pipe( | ||||
|       transform(function (data, file) { | ||||
|         return lokaliseTransform(data, data, file); | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) | ||||
|     .pipe( | ||||
|       merge({ | ||||
|         fileName: "translationMaster.json", | ||||
| @@ -198,18 +176,14 @@ gulp.task("build-master-translation", function () { | ||||
|     .pipe(gulp.dest(workDir)); | ||||
| }); | ||||
|  | ||||
| gulp.task("build-merged-translations", function () { | ||||
|   return gulp | ||||
| gulp.task("build-merged-translations", () => | ||||
|   gulp | ||||
|     .src([inFrontendDir + "/*.json", workDir + "/test.json"], { | ||||
|       allowEmpty: true, | ||||
|     }) | ||||
|     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) | ||||
|     .pipe( | ||||
|       transform(function (data, file) { | ||||
|         return lokaliseTransform(data, data, file); | ||||
|       }) | ||||
|     ) | ||||
|     .pipe( | ||||
|       foreach(function (stream, file) { | ||||
|       foreach((stream, file) => { | ||||
|         // For each language generate a merged json file. It begins with the master | ||||
|         // translation as a failsafe for untranslated strings, and merges all parent | ||||
|         // tags into one file for each specific subtag | ||||
| @@ -241,17 +215,17 @@ gulp.task("build-merged-translations", function () { | ||||
|           ) | ||||
|           .pipe(gulp.dest(fullDir)); | ||||
|       }) | ||||
|     ); | ||||
| }); | ||||
|     ) | ||||
| ); | ||||
|  | ||||
| var taskName; | ||||
| let taskName; | ||||
|  | ||||
| const splitTasks = []; | ||||
| TRANSLATION_FRAGMENTS.forEach((fragment) => { | ||||
|   taskName = "build-translation-fragment-" + fragment; | ||||
|   gulp.task(taskName, function () { | ||||
|   gulp.task(taskName, () => | ||||
|     // Return only the translations for this fragment. | ||||
|     return gulp | ||||
|     gulp | ||||
|       .src(fullDir + "/*.json") | ||||
|       .pipe( | ||||
|         transform((data) => ({ | ||||
| @@ -262,32 +236,33 @@ TRANSLATION_FRAGMENTS.forEach((fragment) => { | ||||
|           }, | ||||
|         })) | ||||
|       ) | ||||
|       .pipe(gulp.dest(workDir + "/" + fragment)); | ||||
|   }); | ||||
|       .pipe(gulp.dest(workDir + "/" + fragment)) | ||||
|   ); | ||||
|   splitTasks.push(taskName); | ||||
| }); | ||||
|  | ||||
| taskName = "build-translation-core"; | ||||
| gulp.task(taskName, function () { | ||||
| gulp.task(taskName, () => | ||||
|   // Remove the fragment translations from the core translation. | ||||
|   return gulp | ||||
|   gulp | ||||
|     .src(fullDir + "/*.json") | ||||
|     .pipe( | ||||
|       transform((data, file) => { | ||||
|       transform((data, _file) => { | ||||
|         TRANSLATION_FRAGMENTS.forEach((fragment) => { | ||||
|           delete data.ui.panel[fragment]; | ||||
|         }); | ||||
|         delete data.supervisor; | ||||
|         return data; | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(gulp.dest(coreDir)); | ||||
| }); | ||||
|     .pipe(gulp.dest(coreDir)) | ||||
| ); | ||||
|  | ||||
| splitTasks.push(taskName); | ||||
|  | ||||
| gulp.task("build-flattened-translations", function () { | ||||
| gulp.task("build-flattened-translations", () => | ||||
|   // Flatten the split versions of our translations, and move them into outDir | ||||
|   return gulp | ||||
|   gulp | ||||
|     .src( | ||||
|       TRANSLATION_FRAGMENTS.map( | ||||
|         (fragment) => workDir + "/" + fragment + "/*.json" | ||||
| @@ -295,41 +270,45 @@ gulp.task("build-flattened-translations", function () { | ||||
|       { base: workDir } | ||||
|     ) | ||||
|     .pipe( | ||||
|       transform(function (data) { | ||||
|       transform((data) => | ||||
|         // Polymer.AppLocalizeBehavior requires flattened json | ||||
|         return flatten(data); | ||||
|       }) | ||||
|         flatten(data) | ||||
|       ) | ||||
|     ) | ||||
|     .pipe( | ||||
|       rename((filePath) => { | ||||
|         if (filePath.dirname === "core") { | ||||
|           filePath.dirname = ""; | ||||
|         } | ||||
|         // In dev we create the file with the fake hash in the filename | ||||
|         if (!env.isProdBuild()) { | ||||
|           filePath.basename += "-dev"; | ||||
|         } | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(gulp.dest(outDir)); | ||||
| }); | ||||
|     .pipe(gulp.dest(outDir)) | ||||
| ); | ||||
|  | ||||
| const fingerprints = {}; | ||||
|  | ||||
| gulp.task( | ||||
|   "build-translation-fingerprints", | ||||
|   function fingerprintTranslationFiles() { | ||||
|     // Fingerprint full file of each language | ||||
|     const files = fs.readdirSync(fullDir); | ||||
| gulp.task("build-translation-fingerprints", () => { | ||||
|   // Fingerprint full file of each language | ||||
|   const files = fs.readdirSync(fullDir); | ||||
|  | ||||
|     for (let i = 0; i < files.length; i++) { | ||||
|       fingerprints[files[i].split(".")[0]] = { | ||||
|         // In dev we create fake hashes | ||||
|         hash: env.isProdBuild() | ||||
|           ? crypto | ||||
|               .createHash("md5") | ||||
|               .update(fs.readFileSync(path.join(fullDir, files[i]), "utf-8")) | ||||
|               .digest("hex") | ||||
|           : "dev", | ||||
|       }; | ||||
|     } | ||||
|   for (let i = 0; i < files.length; i++) { | ||||
|     fingerprints[files[i].split(".")[0]] = { | ||||
|       // In dev we create fake hashes | ||||
|       hash: env.isProdBuild() | ||||
|         ? crypto | ||||
|             .createHash("md5") | ||||
|             .update(fs.readFileSync(path.join(fullDir, files[i]), "utf-8")) | ||||
|             .digest("hex") | ||||
|         : "dev", | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   // In dev we create the file with the fake hash in the filename | ||||
|   if (env.isProdBuild()) { | ||||
|     mapFiles(outDir, ".json", (filename) => { | ||||
|       const parsed = path.parse(filename); | ||||
|  | ||||
| @@ -345,12 +324,88 @@ gulp.task( | ||||
|         }` | ||||
|       ); | ||||
|     }); | ||||
|  | ||||
|     const stream = source("translationFingerprints.json"); | ||||
|     stream.write(JSON.stringify(fingerprints)); | ||||
|     process.nextTick(() => stream.end()); | ||||
|     return stream.pipe(vinylBuffer()).pipe(gulp.dest(workDir)); | ||||
|   } | ||||
|  | ||||
|   const stream = source("translationFingerprints.json"); | ||||
|   stream.write(JSON.stringify(fingerprints)); | ||||
|   process.nextTick(() => stream.end()); | ||||
|   return stream.pipe(vinylBuffer()).pipe(gulp.dest(workDir)); | ||||
| }); | ||||
|  | ||||
| gulp.task("build-translation-fragment-supervisor", () => | ||||
|   gulp | ||||
|     .src(fullDir + "/*.json") | ||||
|     .pipe(transform((data) => data.supervisor)) | ||||
|     .pipe( | ||||
|       rename((filePath) => { | ||||
|         // In dev we create the file with the fake hash in the filename | ||||
|         if (!env.isProdBuild()) { | ||||
|           filePath.basename += "-dev"; | ||||
|         } | ||||
|       }) | ||||
|     ) | ||||
|     .pipe(gulp.dest(workDir + "/supervisor")) | ||||
| ); | ||||
|  | ||||
| gulp.task("build-translation-flatten-supervisor", () => | ||||
|   gulp | ||||
|     .src(workDir + "/supervisor/*.json") | ||||
|     .pipe( | ||||
|       transform((data) => | ||||
|         // Polymer.AppLocalizeBehavior requires flattened json | ||||
|         flatten(data) | ||||
|       ) | ||||
|     ) | ||||
|     .pipe(gulp.dest(outDir)) | ||||
| ); | ||||
|  | ||||
| gulp.task("build-translation-write-metadata", () => | ||||
|   gulp | ||||
|     .src( | ||||
|       [ | ||||
|         path.join(paths.translations_src, "translationMetadata.json"), | ||||
|         workDir + "/testMetadata.json", | ||||
|         workDir + "/translationFingerprints.json", | ||||
|       ], | ||||
|       { allowEmpty: true } | ||||
|     ) | ||||
|     .pipe(merge({})) | ||||
|     .pipe( | ||||
|       transform((data) => { | ||||
|         const newData = {}; | ||||
|         Object.entries(data).forEach(([key, value]) => { | ||||
|           // Filter out translations without native name. | ||||
|           if (value.nativeName) { | ||||
|             newData[key] = value; | ||||
|           } else { | ||||
|             // eslint-disable-next-line no-console | ||||
|             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( | ||||
|   "create-translations", | ||||
|   gulp.series( | ||||
|     env.isProdBuild() ? (done) => done() : "create-test-translation", | ||||
|     "build-master-translation", | ||||
|     "build-merged-translations", | ||||
|     gulp.parallel(...splitTasks), | ||||
|     "build-flattened-translations" | ||||
|   ) | ||||
| ); | ||||
|  | ||||
| gulp.task( | ||||
| @@ -358,48 +413,22 @@ gulp.task( | ||||
|   gulp.series( | ||||
|     "clean-translations", | ||||
|     "ensure-translations-build-dir", | ||||
|     env.isProdBuild() ? (done) => done() : "create-test-translation", | ||||
|     "build-master-translation", | ||||
|     "build-merged-translations", | ||||
|     gulp.parallel(...splitTasks), | ||||
|     "build-flattened-translations", | ||||
|     "create-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,6 @@ | ||||
| /* eslint-disable @typescript-eslint/no-var-requires */ | ||||
| // Tasks to run webpack. | ||||
| const fs = require("fs"); | ||||
| const gulp = require("gulp"); | ||||
| const webpack = require("webpack"); | ||||
| const WebpackDevServer = require("webpack-dev-server"); | ||||
| @@ -18,6 +20,13 @@ const bothBuilds = (createConfigFunc, params) => [ | ||||
|   createConfigFunc({ ...params, latestBuild: false }), | ||||
| ]; | ||||
|  | ||||
| const isWsl = | ||||
|   fs.existsSync("/proc/version") && | ||||
|   fs | ||||
|     .readFileSync("/proc/version", "utf-8") | ||||
|     .toLocaleLowerCase() | ||||
|     .includes("microsoft"); | ||||
|  | ||||
| /** | ||||
|  * @param {{ | ||||
|  *   compiler: import("webpack").Compiler, | ||||
| @@ -36,7 +45,7 @@ const runDevServer = ({ | ||||
|     open: true, | ||||
|     watchContentBase: true, | ||||
|     contentBase, | ||||
|   }).listen(port, listenHost, function (err) { | ||||
|   }).listen(port, listenHost, (err) => { | ||||
|     if (err) { | ||||
|       throw err; | ||||
|     } | ||||
| @@ -47,7 +56,7 @@ const runDevServer = ({ | ||||
|     ); | ||||
|   }); | ||||
|  | ||||
| const handler = (done) => (err, stats) => { | ||||
| const doneHandler = (done) => (err, stats) => { | ||||
|   if (err) { | ||||
|     log.error(err.stack || err); | ||||
|     if (err.details) { | ||||
| @@ -57,6 +66,7 @@ const handler = (done) => (err, stats) => { | ||||
|   } | ||||
|  | ||||
|   if (stats.hasErrors() || stats.hasWarnings()) { | ||||
|     // eslint-disable-next-line no-console | ||||
|     console.log(stats.toString("minimal")); | ||||
|   } | ||||
|  | ||||
| @@ -67,27 +77,34 @@ 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 | ||||
|   webpack(createAppConfig({ isProdBuild: false, latestBuild: true })).watch( | ||||
|     { ignored: /build-translations/ }, | ||||
|     handler() | ||||
|   ); | ||||
|   // This command will run forever because we don't close compiler | ||||
|   webpack( | ||||
|     process.env.ES5 | ||||
|       ? bothBuilds(createAppConfig, { isProdBuild: false }) | ||||
|       : createAppConfig({ isProdBuild: false, latestBuild: true }) | ||||
|   ).watch({ poll: isWsl }, doneHandler()); | ||||
|   gulp.watch( | ||||
|     path.join(paths.translations_src, "en.json"), | ||||
|     gulp.series("build-translations", "copy-translations-app") | ||||
|     gulp.series("create-translations", "copy-translations-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 +115,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 +133,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 +173,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" | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -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,10 +1,12 @@ | ||||
| /* 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"); | ||||
| const WebpackBar = require("webpackbar"); | ||||
|  | ||||
| class LogStartCompilePlugin { | ||||
|   ignoredFirst = false; | ||||
| @@ -36,6 +38,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,15 +48,18 @@ const createWebpackConfig = ({ | ||||
|       rules: [ | ||||
|         { | ||||
|           test: /\.m?js$|\.ts$/, | ||||
|           exclude: bundle.babelExclude(), | ||||
|           use: { | ||||
|             loader: "babel-loader", | ||||
|             options: bundle.babelOptions({ latestBuild }), | ||||
|             options: { | ||||
|               ...bundle.babelOptions({ latestBuild }), | ||||
|               cacheDirectory: !isProdBuild, | ||||
|               cacheCompression: false, | ||||
|             }, | ||||
|           }, | ||||
|         }, | ||||
|         { | ||||
|           test: /\.css$/, | ||||
|           use: "raw-loader", | ||||
|           type: "asset/source", | ||||
|         }, | ||||
|       ], | ||||
|     }, | ||||
| @@ -65,9 +71,12 @@ const createWebpackConfig = ({ | ||||
|           terserOptions: bundle.terserOptions(latestBuild), | ||||
|         }), | ||||
|       ], | ||||
|       moduleIds: isProdBuild && !isStatsBuild ? "deterministic" : "named", | ||||
|       chunkIds: isProdBuild && !isStatsBuild ? "deterministic" : "named", | ||||
|     }, | ||||
|     plugins: [ | ||||
|       new ManifestPlugin({ | ||||
|       new WebpackBar({ fancy: !isProdBuild }), | ||||
|       new WebpackManifestPlugin({ | ||||
|         // Only include the JS of entrypoints | ||||
|         filter: (file) => file.isInitial && !file.name.endsWith(".map"), | ||||
|       }), | ||||
| @@ -93,6 +102,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, | ||||
| @@ -110,82 +120,61 @@ const createWebpackConfig = ({ | ||||
|         new RegExp(bundle.emptyPackages({ latestBuild }).join("|")), | ||||
|         path.resolve(paths.polymer_dir, "src/util/empty.js") | ||||
|       ), | ||||
|       // 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/EventTarget-ponyfill.js") | ||||
|       ), | ||||
|       !isProdBuild && new LogStartCompilePlugin(), | ||||
|     ].filter(Boolean), | ||||
|     resolve: { | ||||
|       extensions: [".ts", ".js", ".json"], | ||||
|       alias: { | ||||
|         "lit/decorators$": "lit/decorators.js", | ||||
|         "lit/directive$": "lit/directive.js", | ||||
|         "lit/directives/until$": "lit/directives/until.js", | ||||
|         "lit/directives/class-map$": "lit/directives/class-map.js", | ||||
|         "lit/directives/style-map$": "lit/directives/style-map.js", | ||||
|         "lit/directives/if-defined$": "lit/directives/if-defined.js", | ||||
|         "lit/directives/guard$": "lit/directives/guard.js", | ||||
|         "lit/directives/cache$": "lit/directives/cache.js", | ||||
|         "lit/directives/repeat$": "lit/directives/repeat.js", | ||||
|         "lit/polyfill-support$": "lit/polyfill-support.js", | ||||
|       }, | ||||
|     }, | ||||
|     output: { | ||||
|       filename: ({ chunk }) => { | ||||
|         if (!isProdBuild || dontHash.has(chunk.name)) { | ||||
|         if (!isProdBuild || isStatsBuild || dontHash.has(chunk.name)) { | ||||
|           return `${chunk.name}.js`; | ||||
|         } | ||||
|         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" | ||||
|           : "[name].chunk.js", | ||||
|         isProdBuild && !isStatsBuild ? "[chunkhash:8].js" : "[id].chunk.js", | ||||
|       path: outputPath, | ||||
|       publicPath, | ||||
|       // To silence warning in worker plugin | ||||
|       globalObject: "self", | ||||
|     }, | ||||
|     experiments: { | ||||
|       topLevelAwait: true, | ||||
|     }, | ||||
|   }; | ||||
| }; | ||||
|  | ||||
| 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, | ||||
|   | ||||
| @@ -139,7 +139,7 @@ | ||||
|           Your authentication credentials or Home Assistant url are never sent | ||||
|           to the Cloud. You can validate this behavior in | ||||
|           <a | ||||
|             href="https://github.com/home-assistant/home-assistant-polymer/tree/dev/cast" | ||||
|             href="https://github.com/home-assistant/frontend/tree/dev/cast" | ||||
|             target="_blank" | ||||
|             >the source code</a | ||||
|           >. | ||||
|   | ||||
| @@ -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; | ||||
|  | ||||
| @@ -201,12 +191,12 @@ class HcCast extends LitElement { | ||||
|       } | ||||
|       this.connection.close(); | ||||
|       location.reload(); | ||||
|     } catch (err) { | ||||
|     } catch (err: any) { | ||||
|       alert("Unable to log out!"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   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> | ||||
| @@ -221,7 +212,7 @@ export class HcConnect extends LitElement { | ||||
|     let url: URL; | ||||
|     try { | ||||
|       url = new URL(value); | ||||
|     } catch (err) { | ||||
|     } catch (err: any) { | ||||
|       this.error = "Invalid URL"; | ||||
|       return; | ||||
|     } | ||||
| @@ -249,7 +240,7 @@ export class HcConnect extends LitElement { | ||||
|     try { | ||||
|       this.loading = true; | ||||
|       auth = await getAuth(options); | ||||
|     } catch (err) { | ||||
|     } catch (err: any) { | ||||
|       if (init === "saved-tokens" && err === ERR_CANNOT_CONNECT) { | ||||
|         this.cannotConnect = true; | ||||
|         return; | ||||
| @@ -268,7 +259,7 @@ export class HcConnect extends LitElement { | ||||
|  | ||||
|     try { | ||||
|       conn = await createConnection({ auth }); | ||||
|     } catch (err) { | ||||
|     } catch (err: any) { | ||||
|       // In case of saved tokens, silently solve problems. | ||||
|       if (init === "saved-tokens") { | ||||
|         if (err === ERR_CANNOT_CONNECT) { | ||||
| @@ -294,12 +285,12 @@ export class HcConnect extends LitElement { | ||||
|     try { | ||||
|       saveTokens(null); | ||||
|       location.reload(); | ||||
|     } catch (err) { | ||||
|     } catch (err: any) { | ||||
|       alert("Unable to log out!"); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   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); | ||||
|   | ||||
| @@ -5,8 +5,8 @@ import { | ||||
| import { castContext } from "../cast_context"; | ||||
|  | ||||
| export const castDemoLovelace: () => LovelaceConfig = () => { | ||||
|   const touchSupported = castContext.getDeviceCapabilities() | ||||
|     .touch_input_supported; | ||||
|   const touchSupported = | ||||
|     castContext.getDeviceCapabilities().touch_input_supported; | ||||
|   return { | ||||
|     views: [ | ||||
|       { | ||||
|   | ||||
| @@ -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; | ||||
|  | ||||
| @@ -152,14 +148,14 @@ export class HcMain extends HassElement { | ||||
|           expires_in: 0, | ||||
|         }), | ||||
|       }); | ||||
|     } catch (err) { | ||||
|     } catch (err: any) { | ||||
|       this._error = this._getErrorMessage(err); | ||||
|       return; | ||||
|     } | ||||
|     let connection; | ||||
|     try { | ||||
|       connection = await createConnection({ auth }); | ||||
|     } catch (err) { | ||||
|     } catch (err: any) { | ||||
|       this._error = this._getErrorMessage(err); | ||||
|       return; | ||||
|     } | ||||
| @@ -197,7 +193,7 @@ export class HcMain extends HassElement { | ||||
|         this._unsubLovelace = llColl.subscribe((lovelaceConfig) => | ||||
|           this._handleNewLovelaceConfig(lovelaceConfig) | ||||
|         ); | ||||
|       } catch (err) { | ||||
|       } catch (err: any) { | ||||
|         // eslint-disable-next-line | ||||
|         console.log("Error fetching Lovelace configuration", err, msg); | ||||
|         // Generate a Lovelace config. | ||||
| @@ -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", | ||||
| @@ -28,6 +29,11 @@ export const demoLovelaceArsaboo: DemoConfig["lovelace"] = (localize) => ({ | ||||
|             }, | ||||
|           ], | ||||
|         }, | ||||
|         { | ||||
|           title: "Energy distribution today", | ||||
|           type: "energy-distribution", | ||||
|           link_dashboard: true, | ||||
|         }, | ||||
|         { | ||||
|           type: "thermostat", | ||||
|           entity: "climate.upstairs", | ||||
| @@ -112,8 +118,7 @@ export const demoLovelaceArsaboo: DemoConfig["lovelace"] = (localize) => ({ | ||||
|                 on: "/assets/arsaboo/icons/light_bulb_on.png", | ||||
|               }, | ||||
|               state_filter: { | ||||
|                 on: | ||||
|                   "brightness(130%) saturate(1.5) drop-shadow(0px 0px 10px gold)", | ||||
|                 on: "brightness(130%) saturate(1.5) drop-shadow(0px 0px 10px gold)", | ||||
|                 off: "brightness(80%) saturate(0.8)", | ||||
|               }, | ||||
|               style: { | ||||
| @@ -195,8 +200,7 @@ export const demoLovelaceArsaboo: DemoConfig["lovelace"] = (localize) => ({ | ||||
|                 on: "/assets/arsaboo/icons/light_bulb_on.png", | ||||
|               }, | ||||
|               state_filter: { | ||||
|                 on: | ||||
|                   "brightness(130%) saturate(1.5) drop-shadow(0px 0px 10px gold)", | ||||
|                 on: "brightness(130%) saturate(1.5) drop-shadow(0px 0px 10px gold)", | ||||
|                 off: "brightness(80%) saturate(0.8)", | ||||
|               }, | ||||
|               style: { | ||||
| @@ -276,8 +280,7 @@ export const demoLovelaceArsaboo: DemoConfig["lovelace"] = (localize) => ({ | ||||
|                 on: "/assets/arsaboo/icons/light_bulb_on.png", | ||||
|               }, | ||||
|               state_filter: { | ||||
|                 on: | ||||
|                   "brightness(130%) saturate(1.5) drop-shadow(0px 0px 10px gold)", | ||||
|                 on: "brightness(130%) saturate(1.5) drop-shadow(0px 0px 10px gold)", | ||||
|                 off: "brightness(80%) saturate(0.8)", | ||||
|               }, | ||||
|               style: { | ||||
| @@ -314,8 +317,7 @@ export const demoLovelaceArsaboo: DemoConfig["lovelace"] = (localize) => ({ | ||||
|                 on: "/assets/arsaboo/icons/light_bulb_on.png", | ||||
|               }, | ||||
|               state_filter: { | ||||
|                 on: | ||||
|                   "brightness(130%) saturate(1.5) drop-shadow(0px 0px 10px gold)", | ||||
|                 on: "brightness(130%) saturate(1.5) drop-shadow(0px 0px 10px gold)", | ||||
|                 off: "brightness(80%) saturate(0.8)", | ||||
|               }, | ||||
|               style: { | ||||
|   | ||||
| @@ -1,32 +1,20 @@ | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
| import { Lovelace } from "../../../src/panels/lovelace/types"; | ||||
| import { energyEntities } from "../stubs/entities"; | ||||
| 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 | ||||
| export let selectedDemoConfigIndex = 0; | ||||
| // eslint-disable-next-line import/no-mutable-exports | ||||
| export let selectedDemoConfig: Promise<DemoConfig> = demoConfigs[ | ||||
|   selectedDemoConfigIndex | ||||
| ](); | ||||
| export let selectedDemoConfig: Promise<DemoConfig> = | ||||
|   demoConfigs[selectedDemoConfigIndex](); | ||||
|  | ||||
| export const setDemoConfig = async ( | ||||
|   hass: MockHomeAssistant, | ||||
| @@ -40,6 +28,7 @@ export const setDemoConfig = async ( | ||||
|   selectedDemoConfig = confProm; | ||||
|  | ||||
|   hass.addEntities(config.entities(hass.localize), true); | ||||
|   hass.addEntities(energyEntities()); | ||||
|   lovelace.saveConfig(config.lovelace(hass.localize)); | ||||
|   hass.mockTheme(config.theme()); | ||||
| }; | ||||
|   | ||||
| @@ -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", | ||||
|         }, | ||||
|   | ||||
| @@ -980,8 +980,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () => | ||||
|         icon: "mdi:account-off", | ||||
|         custom_ui_state_card: "state-card-custom-ui", | ||||
|         templates: { | ||||
|           icon: | ||||
|             "if (state === 'on') return 'mdi:account'; else if (state === 'off') return 'mdi:account-off';\n", | ||||
|           icon: "if (state === 'on') return 'mdi:account'; else if (state === 'off') return 'mdi:account-off';\n", | ||||
|           icon_color: | ||||
|             "if (state === 'on') return 'rgb(56, 150, 56)'; else if (state === 'off') return 'rgb(249, 251, 255)';\n", | ||||
|         }, | ||||
| @@ -1005,8 +1004,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () => | ||||
|         icon: "mdi:account-multiple-minus", | ||||
|         custom_ui_state_card: "state-card-custom-ui", | ||||
|         templates: { | ||||
|           icon: | ||||
|             "if (state === 'on') return 'mdi:account-group'; else if (state === 'off') return 'mdi:account-multiple-minus';\n", | ||||
|           icon: "if (state === 'on') return 'mdi:account-group'; else if (state === 'off') return 'mdi:account-multiple-minus';\n", | ||||
|           icon_color: | ||||
|             "if (state === 'on') return 'rgb(56, 150, 56)'; else if (state === 'off') return 'rgb(249, 251, 255)';\n", | ||||
|         }, | ||||
| @@ -1114,6 +1112,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 +1126,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 +1138,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 +1152,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 +1313,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 +1327,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 +1343,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 +1359,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 +1374,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 +1390,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 +1401,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 +1768,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 +1784,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 +1800,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 +1817,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 +1862,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 = false; | ||||
|  | ||||
|   private _hidden = localStorage.hide_demo_card; | ||||
|  | ||||
| @@ -34,12 +27,7 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|     return this._hidden ? 0 : 2; | ||||
|   } | ||||
|  | ||||
|   public setConfig( | ||||
|     // @ts-ignore | ||||
|     // eslint-disable-next-line @typescript-eslint/no-unused-vars | ||||
|     config: LovelaceCardConfig | ||||
|     // eslint-disable-next-line @typescript-eslint/no-empty-function | ||||
|   ) {} | ||||
|   public setConfig(_config: LovelaceCardConfig) {} | ||||
|  | ||||
|   protected render(): TemplateResult { | ||||
|     if (this._hidden) { | ||||
| @@ -56,7 +44,7 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|                     (conf) => html` | ||||
|                       ${conf.name} | ||||
|                       <small> | ||||
|                         <a target="_blank" href="${conf.authorUrl}"> | ||||
|                         <a target="_blank" href=${conf.authorUrl}> | ||||
|                           ${this.hass.localize( | ||||
|                             "ui.panel.page-demo.cards.demo.demo_by", | ||||
|                             "name", | ||||
| @@ -106,14 +94,14 @@ export class HADemoCard extends LitElement implements LovelaceCard { | ||||
|     this._switching = true; | ||||
|     try { | ||||
|       await setDemoConfig(this.hass, this.lovelace!, index); | ||||
|     } catch (err) { | ||||
|     } catch (err: any) { | ||||
|       alert("Failed to switch config :-("); | ||||
|     } finally { | ||||
|       this._switching = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   static get styles(): CSSResult[] { | ||||
|   static get styles(): CSSResultGroup { | ||||
|     return [ | ||||
|       css` | ||||
|         a { | ||||
|   | ||||
| @@ -1,13 +1,9 @@ | ||||
| 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"; | ||||
| @@ -19,11 +20,14 @@ import { mockShoppingList } from "./stubs/shopping_list"; | ||||
| import { mockSystemLog } from "./stubs/system_log"; | ||||
| import { mockTemplate } from "./stubs/template"; | ||||
| import { mockTranslations } from "./stubs/translations"; | ||||
| import { mockEnergy } from "./stubs/energy"; | ||||
| import { mockConfig } from "./stubs/config"; | ||||
| import { energyEntities } from "./stubs/entities"; | ||||
|  | ||||
| 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), | ||||
| @@ -46,8 +50,12 @@ class HaDemo extends HomeAssistantAppEl { | ||||
|     mockEvents(hass); | ||||
|     mockMediaPlayer(hass); | ||||
|     mockFrontend(hass); | ||||
|     mockEnergy(hass); | ||||
|     mockConfig(hass); | ||||
|     mockPersistentNotification(hass); | ||||
|  | ||||
|     hass.addEntities(energyEntities()); | ||||
|  | ||||
|     // Once config is loaded AND localize, set entities and apply theme. | ||||
|     Promise.all([selectedDemoConfig, localizePromise]).then( | ||||
|       ([conf, localize]) => { | ||||
| @@ -69,7 +77,7 @@ class HaDemo extends HomeAssistantAppEl { | ||||
|         } | ||||
|  | ||||
|         e.preventDefault(); | ||||
|         navigate(this, href); | ||||
|         navigate(href); | ||||
|       }, | ||||
|       { capture: true } | ||||
|     ); | ||||
|   | ||||
							
								
								
									
										41
									
								
								demo/src/stubs/config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								demo/src/stubs/config.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockConfig = (hass: MockHomeAssistant) => { | ||||
|   hass.mockAPI("config/config_entries/entry", () => [ | ||||
|     { | ||||
|       entry_id: "co2signal", | ||||
|       domain: "co2signal", | ||||
|       title: "CO2 Signal", | ||||
|       source: "user", | ||||
|       state: "loaded", | ||||
|       supports_options: false, | ||||
|       supports_unload: true, | ||||
|       pref_disable_new_entities: false, | ||||
|       pref_disable_polling: false, | ||||
|       disabled_by: null, | ||||
|       reason: null, | ||||
|     }, | ||||
|   ]); | ||||
|   hass.mockWS("config/entity_registry/list", () => [ | ||||
|     { | ||||
|       config_entry_id: "co2signal", | ||||
|       device_id: "co2signal", | ||||
|       area_id: null, | ||||
|       disabled_by: null, | ||||
|       entity_id: "sensor.co2_intensity", | ||||
|       name: null, | ||||
|       icon: null, | ||||
|       platform: "co2signal", | ||||
|     }, | ||||
|     { | ||||
|       config_entry_id: "co2signal", | ||||
|       device_id: "co2signal", | ||||
|       area_id: null, | ||||
|       disabled_by: null, | ||||
|       entity_id: "sensor.grid_fossil_fuel_percentage", | ||||
|       name: null, | ||||
|       icon: null, | ||||
|       platform: "co2signal", | ||||
|     }, | ||||
|   ]); | ||||
| }; | ||||
							
								
								
									
										134
									
								
								demo/src/stubs/energy.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										134
									
								
								demo/src/stubs/energy.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,134 @@ | ||||
| import { format, startOfToday, startOfTomorrow } from "date-fns"; | ||||
| import { EnergySolarForecasts } from "../../../src/data/energy"; | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| export const mockEnergy = (hass: MockHomeAssistant) => { | ||||
|   hass.mockWS("energy/get_prefs", () => ({ | ||||
|     energy_sources: [ | ||||
|       { | ||||
|         type: "grid", | ||||
|         flow_from: [ | ||||
|           { | ||||
|             stat_energy_from: "sensor.energy_consumption_tarif_1", | ||||
|             stat_cost: "sensor.energy_consumption_tarif_1_cost", | ||||
|             entity_energy_from: "sensor.energy_consumption_tarif_1", | ||||
|             entity_energy_price: null, | ||||
|             number_energy_price: null, | ||||
|           }, | ||||
|           { | ||||
|             stat_energy_from: "sensor.energy_consumption_tarif_2", | ||||
|             stat_cost: "sensor.energy_consumption_tarif_2_cost", | ||||
|             entity_energy_from: "sensor.energy_consumption_tarif_2", | ||||
|             entity_energy_price: null, | ||||
|             number_energy_price: null, | ||||
|           }, | ||||
|         ], | ||||
|         flow_to: [ | ||||
|           { | ||||
|             stat_energy_to: "sensor.energy_production_tarif_1", | ||||
|             stat_compensation: "sensor.energy_production_tarif_1_compensation", | ||||
|             entity_energy_to: "sensor.energy_production_tarif_1", | ||||
|             entity_energy_price: null, | ||||
|             number_energy_price: null, | ||||
|           }, | ||||
|           { | ||||
|             stat_energy_to: "sensor.energy_production_tarif_2", | ||||
|             stat_compensation: "sensor.energy_production_tarif_2_compensation", | ||||
|             entity_energy_to: "sensor.energy_production_tarif_2", | ||||
|             entity_energy_price: null, | ||||
|             number_energy_price: null, | ||||
|           }, | ||||
|         ], | ||||
|         cost_adjustment_day: 0, | ||||
|       }, | ||||
|       { | ||||
|         type: "solar", | ||||
|         stat_energy_from: "sensor.solar_production", | ||||
|         config_entry_solar_forecast: ["solar_forecast"], | ||||
|       }, | ||||
|       /* { | ||||
|         type: "battery", | ||||
|         stat_energy_from: "sensor.battery_output", | ||||
|         stat_energy_to: "sensor.battery_input", | ||||
|       }, */ | ||||
|       { | ||||
|         type: "gas", | ||||
|         stat_energy_from: "sensor.energy_gas", | ||||
|         stat_cost: "sensor.energy_gas_cost", | ||||
|         entity_energy_from: "sensor.energy_gas", | ||||
|         entity_energy_price: null, | ||||
|         number_energy_price: null, | ||||
|       }, | ||||
|     ], | ||||
|     device_consumption: [ | ||||
|       { | ||||
|         stat_consumption: "sensor.energy_car", | ||||
|       }, | ||||
|       { | ||||
|         stat_consumption: "sensor.energy_ac", | ||||
|       }, | ||||
|       { | ||||
|         stat_consumption: "sensor.energy_washing_machine", | ||||
|       }, | ||||
|       { | ||||
|         stat_consumption: "sensor.energy_dryer", | ||||
|       }, | ||||
|       { | ||||
|         stat_consumption: "sensor.energy_heat_pump", | ||||
|       }, | ||||
|       { | ||||
|         stat_consumption: "sensor.energy_boiler", | ||||
|       }, | ||||
|     ], | ||||
|   })); | ||||
|   hass.mockWS("energy/info", () => ({ cost_sensors: [] })); | ||||
|   const todayString = format(startOfToday(), "yyyy-MM-dd"); | ||||
|   const tomorrowString = format(startOfTomorrow(), "yyyy-MM-dd"); | ||||
|   hass.mockWS( | ||||
|     "energy/solar_forecast", | ||||
|     (): EnergySolarForecasts => ({ | ||||
|       solar_forecast: { | ||||
|         wh_hours: { | ||||
|           [`${todayString}T06:00:00`]: 0, | ||||
|           [`${todayString}T06:23:00`]: 6, | ||||
|           [`${todayString}T06:45:00`]: 39, | ||||
|           [`${todayString}T07:00:00`]: 28, | ||||
|           [`${todayString}T08:00:00`]: 208, | ||||
|           [`${todayString}T09:00:00`]: 352, | ||||
|           [`${todayString}T10:00:00`]: 544, | ||||
|           [`${todayString}T11:00:00`]: 748, | ||||
|           [`${todayString}T12:00:00`]: 1259, | ||||
|           [`${todayString}T13:00:00`]: 1361, | ||||
|           [`${todayString}T14:00:00`]: 1373, | ||||
|           [`${todayString}T15:00:00`]: 1370, | ||||
|           [`${todayString}T16:00:00`]: 1186, | ||||
|           [`${todayString}T17:00:00`]: 937, | ||||
|           [`${todayString}T18:00:00`]: 652, | ||||
|           [`${todayString}T19:00:00`]: 370, | ||||
|           [`${todayString}T20:00:00`]: 155, | ||||
|           [`${todayString}T21:48:00`]: 24, | ||||
|           [`${todayString}T22:36:00`]: 0, | ||||
|           [`${tomorrowString}T06:01:00`]: 0, | ||||
|           [`${tomorrowString}T06:23:00`]: 9, | ||||
|           [`${tomorrowString}T06:45:00`]: 47, | ||||
|           [`${tomorrowString}T07:00:00`]: 48, | ||||
|           [`${tomorrowString}T08:00:00`]: 473, | ||||
|           [`${tomorrowString}T09:00:00`]: 827, | ||||
|           [`${tomorrowString}T10:00:00`]: 1153, | ||||
|           [`${tomorrowString}T11:00:00`]: 1413, | ||||
|           [`${tomorrowString}T12:00:00`]: 1590, | ||||
|           [`${tomorrowString}T13:00:00`]: 1652, | ||||
|           [`${tomorrowString}T14:00:00`]: 1612, | ||||
|           [`${tomorrowString}T15:00:00`]: 1438, | ||||
|           [`${tomorrowString}T16:00:00`]: 1149, | ||||
|           [`${tomorrowString}T17:00:00`]: 830, | ||||
|           [`${tomorrowString}T18:00:00`]: 542, | ||||
|           [`${tomorrowString}T19:00:00`]: 311, | ||||
|           [`${tomorrowString}T20:00:00`]: 140, | ||||
|           [`${tomorrowString}T21:47:00`]: 22, | ||||
|           [`${tomorrowString}T22:34:00`]: 0, | ||||
|         }, | ||||
|       }, | ||||
|     }) | ||||
|   ); | ||||
| }; | ||||
							
								
								
									
										178
									
								
								demo/src/stubs/entities.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								demo/src/stubs/entities.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,178 @@ | ||||
| import { convertEntities } from "../../../src/fake_data/entity"; | ||||
|  | ||||
| export const energyEntities = () => | ||||
|   convertEntities({ | ||||
|     "sensor.grid_fossil_fuel_percentage": { | ||||
|       entity_id: "sensor.grid_fossil_fuel_percentage", | ||||
|       state: "88.6", | ||||
|       attributes: { | ||||
|         unit_of_measurement: "%", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.solar_production": { | ||||
|       entity_id: "sensor.solar_production", | ||||
|       state: "88.6", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Solar", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.battery_input": { | ||||
|       entity_id: "sensor.battery_input", | ||||
|       state: "4", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Battery Input", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.battery_output": { | ||||
|       entity_id: "sensor.battery_output", | ||||
|       state: "3", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Battery Output", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_consumption_tarif_1": { | ||||
|       entity_id: "sensor.energy_consumption_tarif_1	", | ||||
|       state: "88.6", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Grid consumption low tariff", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_consumption_tarif_2": { | ||||
|       entity_id: "sensor.energy_consumption_tarif_2", | ||||
|       state: "88.6", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Grid consumption high tariff", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_production_tarif_1": { | ||||
|       entity_id: "sensor.energy_production_tarif_1", | ||||
|       state: "88.6", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Returned to grid low tariff", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_production_tarif_2": { | ||||
|       entity_id: "sensor.energy_production_tarif_2", | ||||
|       state: "88.6", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Returned to grid high tariff", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_consumption_tarif_1_cost": { | ||||
|       entity_id: "sensor.energy_consumption_tarif_1_cost", | ||||
|       state: "2", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         unit_of_measurement: "EUR", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_consumption_tarif_2_cost": { | ||||
|       entity_id: "sensor.energy_consumption_tarif_2_cost", | ||||
|       state: "2", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         unit_of_measurement: "EUR", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_production_tarif_1_compensation": { | ||||
|       entity_id: "sensor.energy_production_tarif_1_compensation", | ||||
|       state: "2", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         unit_of_measurement: "EUR", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_production_tarif_2_compensation": { | ||||
|       entity_id: "sensor.energy_production_tarif_2_compensation", | ||||
|       state: "2", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         unit_of_measurement: "EUR", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_gas_cost": { | ||||
|       entity_id: "sensor.energy_gas_cost", | ||||
|       state: "2", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         unit_of_measurement: "EUR", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_gas": { | ||||
|       entity_id: "sensor.energy_gas", | ||||
|       state: "4", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Gas", | ||||
|         unit_of_measurement: "m³", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_car": { | ||||
|       entity_id: "sensor.energy_car", | ||||
|       state: "4", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Electric car", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_ac": { | ||||
|       entity_id: "sensor.energy_ac", | ||||
|       state: "3", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Air conditioning", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_washing_machine": { | ||||
|       entity_id: "sensor.energy_washing_machine", | ||||
|       state: "6", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Washing machine", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_dryer": { | ||||
|       entity_id: "sensor.energy_dryer", | ||||
|       state: "5.5", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Dryer", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_heat_pump": { | ||||
|       entity_id: "sensor.energy_heat_pump", | ||||
|       state: "6", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Heat pump", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|     "sensor.energy_boiler": { | ||||
|       entity_id: "sensor.energy_boiler", | ||||
|       state: "7", | ||||
|       attributes: { | ||||
|         last_reset: "1970-01-01T00:00:00:00+00", | ||||
|         friendly_name: "Boiler", | ||||
|         unit_of_measurement: "kWh", | ||||
|       }, | ||||
|     }, | ||||
|   }); | ||||
| @@ -1,4 +1,6 @@ | ||||
| import { addHours, differenceInHours, endOfDay } from "date-fns"; | ||||
| import { HassEntity } from "home-assistant-js-websocket"; | ||||
| import { StatisticValue } from "../../../src/data/history"; | ||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||
|  | ||||
| interface HistoryQueryParams { | ||||
| @@ -64,17 +66,219 @@ const generateHistory = (state, deltas) => { | ||||
|  | ||||
| const incrementalUnits = ["clients", "queries", "ads"]; | ||||
|  | ||||
| const generateMeanStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| ) => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let lastVal = initValue; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const delta = Math.random() * maxDiff; | ||||
|     const mean = lastVal + delta; | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       mean, | ||||
|       min: mean - Math.random() * maxDiff, | ||||
|       max: mean + Math.random() * maxDiff, | ||||
|       last_reset: "1970-01-01T00:00:00+00:00", | ||||
|       state: mean, | ||||
|       sum: null, | ||||
|     }); | ||||
|     lastVal = mean; | ||||
|     currentDate = addHours(currentDate, 1); | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const generateSumStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   initValue: number, | ||||
|   maxDiff: number | ||||
| ) => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let sum = initValue; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const add = Math.random() * maxDiff; | ||||
|     sum += add; | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       mean: null, | ||||
|       min: null, | ||||
|       max: null, | ||||
|       last_reset: "1970-01-01T00:00:00+00:00", | ||||
|       state: initValue + sum, | ||||
|       sum, | ||||
|     }); | ||||
|     currentDate = addHours(currentDate, 1); | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const generateCurvedStatistics = ( | ||||
|   id: string, | ||||
|   start: Date, | ||||
|   end: Date, | ||||
|   initValue: number, | ||||
|   maxDiff: number, | ||||
|   metered: boolean | ||||
| ) => { | ||||
|   const statistics: StatisticValue[] = []; | ||||
|   let currentDate = new Date(start); | ||||
|   currentDate.setMinutes(0, 0, 0); | ||||
|   let sum = initValue; | ||||
|   const hours = differenceInHours(end, start) - 1; | ||||
|   let i = 0; | ||||
|   let half = false; | ||||
|   const now = new Date(); | ||||
|   while (end > currentDate && currentDate < now) { | ||||
|     const add = Math.random() * maxDiff; | ||||
|     sum += i * add; | ||||
|     statistics.push({ | ||||
|       statistic_id: id, | ||||
|       start: currentDate.toISOString(), | ||||
|       mean: null, | ||||
|       min: null, | ||||
|       max: null, | ||||
|       last_reset: "1970-01-01T00:00:00+00:00", | ||||
|       state: initValue + sum, | ||||
|       sum: metered ? sum : null, | ||||
|     }); | ||||
|     currentDate = addHours(currentDate, 1); | ||||
|     if (!half && i > hours / 2) { | ||||
|       half = true; | ||||
|     } | ||||
|     i += half ? -1 : 1; | ||||
|   } | ||||
|   return statistics; | ||||
| }; | ||||
|  | ||||
| const statisticsFunctions: Record< | ||||
|   string, | ||||
|   (id: string, start: Date, end: Date) => StatisticValue[] | ||||
| > = { | ||||
|   "sensor.energy_consumption_tarif_1": (id: string, start: Date, end: Date) => { | ||||
|     const morningEnd = new Date(start.getTime() + 10 * 60 * 60 * 1000); | ||||
|     const morningLow = generateSumStatistics(id, start, morningEnd, 0, 0.7); | ||||
|     const eveningStart = new Date(start.getTime() + 20 * 60 * 60 * 1000); | ||||
|     const morningFinalVal = morningLow.length | ||||
|       ? morningLow[morningLow.length - 1].sum! | ||||
|       : 0; | ||||
|     const empty = generateSumStatistics( | ||||
|       id, | ||||
|       morningEnd, | ||||
|       eveningStart, | ||||
|       morningFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const eveningLow = generateSumStatistics( | ||||
|       id, | ||||
|       eveningStart, | ||||
|       end, | ||||
|       morningFinalVal, | ||||
|       0.7 | ||||
|     ); | ||||
|     return [...morningLow, ...empty, ...eveningLow]; | ||||
|   }, | ||||
|   "sensor.energy_consumption_tarif_2": (id: string, start: Date, end: Date) => { | ||||
|     const morningEnd = new Date(start.getTime() + 9 * 60 * 60 * 1000); | ||||
|     const eveningStart = new Date(start.getTime() + 20 * 60 * 60 * 1000); | ||||
|     const highTarif = generateSumStatistics( | ||||
|       id, | ||||
|       morningEnd, | ||||
|       eveningStart, | ||||
|       0, | ||||
|       0.3 | ||||
|     ); | ||||
|     const highTarifFinalVal = highTarif.length | ||||
|       ? highTarif[highTarif.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(id, start, morningEnd, 0, 0); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       eveningStart, | ||||
|       end, | ||||
|       highTarifFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     return [...morning, ...highTarif, ...evening]; | ||||
|   }, | ||||
|   "sensor.energy_production_tarif_1": (id, start, end) => | ||||
|     generateSumStatistics(id, start, end, 0, 0), | ||||
|   "sensor.energy_production_tarif_1_compensation": (id, start, end) => | ||||
|     generateSumStatistics(id, start, end, 0, 0), | ||||
|   "sensor.energy_production_tarif_2": (id, start, end) => { | ||||
|     const productionStart = new Date(start.getTime() + 9 * 60 * 60 * 1000); | ||||
|     const productionEnd = new Date(start.getTime() + 21 * 60 * 60 * 1000); | ||||
|     const dayEnd = new Date(endOfDay(productionEnd)); | ||||
|     const production = generateCurvedStatistics( | ||||
|       id, | ||||
|       productionStart, | ||||
|       productionEnd, | ||||
|       0, | ||||
|       0.15, | ||||
|       true | ||||
|     ); | ||||
|     const productionFinalVal = production.length | ||||
|       ? production[production.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(id, start, productionStart, 0, 0); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       productionEnd, | ||||
|       dayEnd, | ||||
|       productionFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const rest = generateSumStatistics(id, dayEnd, end, productionFinalVal, 1); | ||||
|     return [...morning, ...production, ...evening, ...rest]; | ||||
|   }, | ||||
|   "sensor.solar_production": (id, start, end) => { | ||||
|     const productionStart = new Date(start.getTime() + 7 * 60 * 60 * 1000); | ||||
|     const productionEnd = new Date(start.getTime() + 23 * 60 * 60 * 1000); | ||||
|     const dayEnd = new Date(endOfDay(productionEnd)); | ||||
|     const production = generateCurvedStatistics( | ||||
|       id, | ||||
|       productionStart, | ||||
|       productionEnd, | ||||
|       0, | ||||
|       0.3, | ||||
|       true | ||||
|     ); | ||||
|     const productionFinalVal = production.length | ||||
|       ? production[production.length - 1].sum! | ||||
|       : 0; | ||||
|     const morning = generateSumStatistics(id, start, productionStart, 0, 0); | ||||
|     const evening = generateSumStatistics( | ||||
|       id, | ||||
|       productionEnd, | ||||
|       dayEnd, | ||||
|       productionFinalVal, | ||||
|       0 | ||||
|     ); | ||||
|     const rest = generateSumStatistics(id, dayEnd, end, productionFinalVal, 2); | ||||
|     return [...morning, ...production, ...evening, ...rest]; | ||||
|   }, | ||||
|   "sensor.grid_fossil_fuel_percentage": (id, start, end) => | ||||
|     generateMeanStatistics(id, start, end, 35, 1.3), | ||||
| }; | ||||
|  | ||||
| export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|   mockHass.mockAPI( | ||||
|     new RegExp("history/period/.+"), | ||||
|     ( | ||||
|       hass, | ||||
|       // @ts-ignore | ||||
|       method, | ||||
|       path, | ||||
|       // @ts-ignore | ||||
|       parameters | ||||
|     ) => { | ||||
|     (hass, _method, path, _parameters) => { | ||||
|       const params = parseQuery<HistoryQueryParams>(path.split("?")[1]); | ||||
|       const entities = params.filter_entity_id.split(","); | ||||
|  | ||||
| @@ -95,7 +299,7 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|         const numberState = Number(state.state); | ||||
|  | ||||
|         if (isNaN(numberState)) { | ||||
|           // eslint-disable-next-line | ||||
|           // eslint-disable-next-line no-console | ||||
|           console.log( | ||||
|             "Ignoring state with unparsable state but with a unit", | ||||
|             entityId, | ||||
| @@ -140,4 +344,40 @@ export const mockHistory = (mockHass: MockHomeAssistant) => { | ||||
|       return results; | ||||
|     } | ||||
|   ); | ||||
|   mockHass.mockWS("history/list_statistic_ids", () => []); | ||||
|   mockHass.mockWS( | ||||
|     "history/statistics_during_period", | ||||
|     ({ statistic_ids, start_time, end_time }, hass) => { | ||||
|       const start = new Date(start_time); | ||||
|       const end = end_time ? new Date(end_time) : new Date(); | ||||
|  | ||||
|       const statistics: Record<string, StatisticValue[]> = {}; | ||||
|  | ||||
|       statistic_ids.forEach((id: string) => { | ||||
|         if (id in statisticsFunctions) { | ||||
|           statistics[id] = statisticsFunctions[id](id, start, end); | ||||
|         } else { | ||||
|           const entityState = hass.states[id]; | ||||
|           const state = entityState ? Number(entityState.state) : 1; | ||||
|           statistics[id] = | ||||
|             entityState && "last_reset" in entityState.attributes | ||||
|               ? generateSumStatistics( | ||||
|                   id, | ||||
|                   start, | ||||
|                   end, | ||||
|                   state, | ||||
|                   state * (state > 80 ? 0.01 : 0.05) | ||||
|                 ) | ||||
|               : generateMeanStatistics( | ||||
|                   id, | ||||
|                   start, | ||||
|                   end, | ||||
|                   state, | ||||
|                   state * (state > 80 ? 0.05 : 0.1) | ||||
|                 ); | ||||
|         } | ||||
|       }); | ||||
|       return statistics; | ||||
|     } | ||||
|   ); | ||||
| }; | ||||
|   | ||||
| @@ -10,10 +10,9 @@ export const mockLovelace = ( | ||||
|   localizePromise: Promise<LocalizeFunc> | ||||
| ) => { | ||||
|   hass.mockWS("lovelace/config", () => | ||||
|     Promise.all([ | ||||
|       selectedDemoConfig, | ||||
|       localizePromise, | ||||
|     ]).then(([config, localize]) => config.lovelace(localize)) | ||||
|     Promise.all([selectedDemoConfig, localizePromise]).then( | ||||
|       ([config, localize]) => config.lovelace(localize) | ||||
|     ) | ||||
|   ); | ||||
|  | ||||
|   hass.mockWS("lovelace/config/save", () => Promise.resolve()); | ||||
| @@ -24,9 +23,9 @@ customElements.whenDefined("hui-view").then(() => { | ||||
|   // eslint-disable-next-line | ||||
|   const HUIView = customElements.get("hui-view"); | ||||
|   // Patch HUI-VIEW to make the lovelace object available to the demo card | ||||
|   const oldCreateCard = HUIView.prototype.createCardElement; | ||||
|   const oldCreateCard = HUIView!.prototype.createCardElement; | ||||
|  | ||||
|   HUIView.prototype.createCardElement = function (config) { | ||||
|   HUIView!.prototype.createCardElement = function (config) { | ||||
|     const el = oldCreateCard.call(this, config); | ||||
|     if (el.tagName === "HA-DEMO-CARD") { | ||||
|       (el as HADemoCard).lovelace = this.lovelace; | ||||
|   | ||||
| @@ -6,7 +6,7 @@ export const mockTemplate = (hass: MockHomeAssistant) => { | ||||
|       body: { message: "Template dev tool does not work in the demo." }, | ||||
|     }) | ||||
|   ); | ||||
|   hass.mockWS("render_template", (msg, onChange) => { | ||||
|   hass.mockWS("render_template", (msg, _hass, onChange) => { | ||||
|     onChange!({ | ||||
|       result: msg.template, | ||||
|       listeners: { all: false, domains: [], entities: [], time: false }, | ||||
|   | ||||
| @@ -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; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										82
									
								
								gallery/src/demos/demo-automation-trace-timeline.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										82
									
								
								gallery/src/demos/demo-automation-trace-timeline.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,82 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement, property } from "lit/decorators"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/trace/hat-script-graph"; | ||||
| import "../../../src/components/trace/hat-trace-timeline"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
| import { mockDemoTrace } from "../data/traces/mock-demo-trace"; | ||||
| import { DemoTrace } from "../data/traces/types"; | ||||
|  | ||||
| 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; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										92
									
								
								gallery/src/demos/demo-automation-trace.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										92
									
								
								gallery/src/demos/demo-automation-trace.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,92 @@ | ||||
| /* eslint-disable lit/no-template-arrow */ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import "../../../src/components/ha-card"; | ||||
| import "../../../src/components/trace/hat-script-graph"; | ||||
| import "../../../src/components/trace/hat-trace-timeline"; | ||||
| import { customElement, property, state } from "lit/decorators"; | ||||
| import { provideHass } from "../../../src/fake_data/provide_hass"; | ||||
| import { HomeAssistant } from "../../../src/types"; | ||||
| import { DemoTrace } from "../data/traces/types"; | ||||
| import { basicTrace } from "../data/traces/basic_trace"; | ||||
| import { motionLightTrace } from "../data/traces/motion-light-trace"; | ||||
|  | ||||
| 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; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										150
									
								
								gallery/src/demos/demo-ha-alert.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								gallery/src/demos/demo-ha-alert.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,150 @@ | ||||
| import { html, css, LitElement, TemplateResult } from "lit"; | ||||
| import { customElement } from "lit/decorators"; | ||||
| import "../../../src/components/ha-alert"; | ||||
| import "../../../src/components/ha-card"; | ||||
|  | ||||
| const alerts: { | ||||
|   title?: string; | ||||
|   description: string | TemplateResult; | ||||
|   type: "info" | "warning" | "error" | "success"; | ||||
|   dismissable?: boolean; | ||||
|   action?: string; | ||||
|   rtl?: boolean; | ||||
| }[] = [ | ||||
|   { | ||||
|     title: "Test info alert", | ||||
|     description: "This is a test info alert with a title and description", | ||||
|     type: "info", | ||||
|   }, | ||||
|   { | ||||
|     title: "Test warning alert", | ||||
|     description: "This is a test warning alert with a title and description", | ||||
|     type: "warning", | ||||
|   }, | ||||
|   { | ||||
|     title: "Test error alert", | ||||
|     description: "This is a test error alert with a title and description", | ||||
|     type: "error", | ||||
|   }, | ||||
|   { | ||||
|     title: "Test warning with long string", | ||||
|     description: | ||||
|       "sensor.lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum_lorem_ipsum", | ||||
|     type: "warning", | ||||
|   }, | ||||
|   { | ||||
|     title: "Test success alert", | ||||
|     description: "This is a test success alert with a title and description", | ||||
|     type: "success", | ||||
|   }, | ||||
|   { | ||||
|     description: "This is a test info alert with description only", | ||||
|     type: "info", | ||||
|   }, | ||||
|   { | ||||
|     description: | ||||
|       "This is a test warning alert with a rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really rally really really long description only", | ||||
|     type: "warning", | ||||
|   }, | ||||
|   { | ||||
|     title: "Error with description and list", | ||||
|     description: html`<p> | ||||
|         This is a test error alert with a title, description and a list | ||||
|       </p> | ||||
|       <ul> | ||||
|         <li>List item #1</li> | ||||
|         <li>List item #2</li> | ||||
|         <li>List item #3</li> | ||||
|       </ul>`, | ||||
|     type: "error", | ||||
|   }, | ||||
|   { | ||||
|     title: "Test dismissable alert", | ||||
|     description: "This is a test success alert that can be dismissable", | ||||
|     type: "success", | ||||
|     dismissable: true, | ||||
|   }, | ||||
|   { | ||||
|     description: "Dismissable information", | ||||
|     type: "info", | ||||
|     dismissable: true, | ||||
|   }, | ||||
|   { | ||||
|     title: "Error with action", | ||||
|     description: "This is a test error alert with action", | ||||
|     type: "error", | ||||
|     action: "restart", | ||||
|   }, | ||||
|   { | ||||
|     title: "Unsaved data", | ||||
|     description: "You have unsaved data", | ||||
|     type: "warning", | ||||
|     action: "save", | ||||
|   }, | ||||
|   { | ||||
|     description: "Dismissable information (RTL)", | ||||
|     type: "info", | ||||
|     dismissable: true, | ||||
|     rtl: true, | ||||
|   }, | ||||
|   { | ||||
|     title: "Error with action", | ||||
|     description: "This is a test error alert with action (RTL)", | ||||
|     type: "error", | ||||
|     action: "restart", | ||||
|     rtl: true, | ||||
|   }, | ||||
|   { | ||||
|     title: "Test success alert (RTL)", | ||||
|     description: "This is a test success alert with a title and description", | ||||
|     type: "success", | ||||
|     rtl: true, | ||||
|   }, | ||||
| ]; | ||||
|  | ||||
| @customElement("demo-ha-alert") | ||||
| export class DemoHaAlert extends LitElement { | ||||
|   protected render(): TemplateResult { | ||||
|     return html` | ||||
|       <ha-card header="ha-alert demo"> | ||||
|         ${alerts.map( | ||||
|           (alert) => html` | ||||
|             <ha-alert | ||||
|               .title=${alert.title || ""} | ||||
|               .alertType=${alert.type} | ||||
|               .dismissable=${alert.dismissable || false} | ||||
|               .actionText=${alert.action || ""} | ||||
|               .rtl=${alert.rtl || false} | ||||
|             > | ||||
|               ${alert.description} | ||||
|             </ha-alert> | ||||
|           ` | ||||
|         )} | ||||
|       </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-ha-alert": DemoHaAlert; | ||||
|   } | ||||
| } | ||||
| @@ -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; | ||||
|   } | ||||
| } | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user