Compare commits
	
		
			1229 Commits
		
	
	
		
			20200519.3
			...
			template-e
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 72bf0c918a | ||
|   | 419f5d13bf | ||
|   | 16549b3404 | ||
|   | cbddebeaa8 | ||
|   | bbe4c95109 | ||
|   | 4c6f9f0dd8 | ||
|   | 90f7dba793 | ||
|   | 7c492338a2 | ||
|   | 530f494df8 | ||
|   | 8fd1f35c59 | ||
|   | af1518e924 | ||
|   | 473e381d75 | ||
|   | 7d3acc747d | ||
|   | bf7424a67c | ||
|   | a856337eae | ||
|   | 6cf47ba4eb | ||
|   | 3b7a189708 | ||
|   | 79c542b76a | ||
|   | e37b7bd73f | ||
|   | d6f3c34b33 | ||
|   | bc5cb46e7d | ||
|   | c7b747c4fa | ||
|   | d3c51d7acd | ||
|   | b6881d797c | ||
|   | b9f802939c | ||
|   | 6558c2c065 | ||
|   | 37a089c868 | ||
|   | f68eff6bb3 | ||
|   | 88a525f1a7 | ||
|   | 34fddd5940 | ||
|   | 0e5d6fe8d8 | ||
|   | e1342a0d9d | ||
|   | 0cc2d3aaa7 | ||
|   | 67814505b3 | ||
|   | bae29c6d62 | ||
|   | a0e67d4c03 | ||
|   | 131bc5fbf7 | ||
|   | 051218e29b | ||
|   | 6ace8307d8 | ||
|   | e84bef44b7 | ||
|   | 3186d762f2 | ||
|   | c97a3b0a56 | ||
|   | 78f1bb3b91 | ||
|   | 67707fbc90 | ||
|   | 2a57ffa615 | ||
|   | 216fce74f8 | ||
|   | 6cd3e6652a | ||
|   | fe7d79cee6 | ||
|   | 2f4e7b388b | ||
|   | 2e289cd152 | ||
|   | 21a3dcf06c | ||
|   | 7f56add914 | ||
|   | 88701c6167 | ||
|   | e4ce6117a1 | ||
|   | cec2a61bdf | ||
|   | 8275ac5853 | ||
|   | b7bcf97365 | ||
|   | fa28b480f1 | ||
|   | 4bb95b7396 | ||
|   | 5a9bd73e8b | ||
|   | 4fe0276914 | ||
|   | 5e8bda55b4 | ||
|   | d09c4898c1 | ||
|   | 6ae67ed299 | ||
|   | 32ff166a74 | ||
|   | 8feae04281 | ||
|   | 129f9c147b | ||
|   | 6e336dd207 | ||
|   | 161561c48a | ||
|   | c162e84383 | ||
|   | dc8d80a6e5 | ||
|   | 293f67968c | ||
|   | 4dcf26236e | ||
|   | a0e8d69243 | ||
|   | 33cd9bf516 | ||
|   | 0132797f2f | ||
|   | 7e2db0aa4e | ||
|   | cc1d50491b | ||
|   | 461b86a04b | ||
|   | 9a3a7c28f4 | ||
|   | 1c9d0200ca | ||
|   | 0037cd2e69 | ||
|   | 028ae061da | ||
|   | 2e47763ecc | ||
|   | 924e4a45d0 | ||
|   | 8361b9553b | ||
|   | e52be20fba | ||
|   | da12233ade | ||
|   | 57500f6c97 | ||
|   | 199e17d0b1 | ||
|   | 3b91343082 | ||
|   | 1753c9163c | ||
|   | 89e5953e89 | ||
|   | 5bfd25c8c6 | ||
|   | e555b24f50 | ||
|   | 14db37459f | ||
|   | 1d9779d47c | ||
|   | 3dedbc5457 | ||
|   | facb3266c6 | ||
|   | 01fe5dd2f7 | ||
|   | 9b22b1e499 | ||
|   | 4bc8818145 | ||
|   | 48ef8c86c2 | ||
|   | 89f359a52f | ||
|   | 13b8160d74 | ||
|   | f1c16d6674 | ||
|   | 76a088e177 | ||
|   | 630d8c3bb6 | ||
|   | 744efa30f2 | ||
|   | bf4a94dc48 | ||
|   | ce4ba2f6f1 | ||
|   | 5b232b5d35 | ||
|   | 35151bbac7 | ||
|   | f0e959319e | ||
|   | d0c4475724 | ||
|   | 99935f1e59 | ||
|   | fbb43821ba | ||
|   | c7f5c6c1d1 | ||
|   | d26f1fa371 | ||
|   | c3718ff7dd | ||
|   | d63493a859 | ||
|   | a72183851a | ||
|   | 40b2387667 | ||
|   | d814aa36a7 | ||
|   | e37eebe4ad | ||
|   | 0baaaefdf8 | ||
|   | 58a58906e7 | ||
|   | bec0d9b00e | ||
|   | e6a4ab789b | ||
|   | 36c1d3230c | ||
|   | 30466ec3fe | ||
|   | ce414a5ca9 | ||
|   | e4e6edd573 | ||
|   | 79927f4dc9 | ||
|   | 603b833757 | ||
|   | ba99d1a10d | ||
|   | efe97e8f51 | ||
|   | 5ec23bb7ab | ||
|   | 9b4d01ab75 | ||
|   | 40191a88d4 | ||
|   | a19477d179 | ||
|   | bf98a78f3d | ||
|   | ba4c2fc1bd | ||
|   | b56e9ef028 | ||
|   | dbbd34c520 | ||
|   | ccb69dbdfa | ||
|   | 11e555ef6f | ||
|   | 61e17395c9 | ||
|   | 733ce3b6b8 | ||
|   | 375f143199 | ||
|   | 2419f35eb9 | ||
|   | 21867c3576 | ||
|   | 28853b28bc | ||
|   | e2f27568a5 | ||
|   | 98b2b796b0 | ||
|   | b8f3fcf00b | ||
|   | d3fda9a821 | ||
|   | 19e69dc13e | ||
|   | 48543a2dad | ||
|   | b22f5ae5c2 | ||
|   | 2acb6a28fe | ||
|   | 1064cdb79d | ||
|   | bd7cb1c877 | ||
|   | 6c314982dc | ||
|   | d54710f113 | ||
|   | 1346156ecd | ||
|   | a2d9f9b417 | ||
|   | 3de78cca2d | ||
|   | 5fa7cd9fa9 | ||
|   | a78c00fb41 | ||
|   | edc2a03d1c | ||
|   | 174f8f5823 | ||
|   | 9fbc94e8d8 | ||
|   | 6aff35196d | ||
|   | eceed4ed74 | ||
|   | 7428731eac | ||
|   | 89b07ea0ae | ||
|   | d16daf0fd9 | ||
|   | 211ab4eea8 | ||
|   | dbd53f8d14 | ||
|   | a27680b8c0 | ||
|   | 07fc9b98cc | ||
|   | 33582c0448 | ||
|   | 73be0fef75 | ||
|   | 611202c905 | ||
|   | e553f35a68 | ||
|   | 673649a603 | ||
|   | c4ed743370 | ||
|   | 682fa0d3eb | ||
|   | 30f34eee22 | ||
|   | eab76bf85b | ||
|   | bcf405bf9d | ||
|   | 3c4b0d4a74 | ||
|   | fb9bd0eb7d | ||
|   | 7e2dc04123 | ||
|   | 54ec37994c | ||
|   | 4a5935ee36 | ||
|   | 01b9a07320 | ||
|   | 0fcf0dcd18 | ||
|   | 80481f142a | ||
|   | 2be08ce7ab | ||
|   | 37eb5af3d4 | ||
|   | 8c8151be92 | ||
|   | baf31d1c1e | ||
|   | af2250835a | ||
|   | 6f2a759ba3 | ||
|   | 5065901196 | ||
|   | 41b59e6e11 | ||
|   | 43afdaadc6 | ||
|   | 83c5151792 | ||
|   | 0880ab67c6 | ||
|   | c0b2143c7c | ||
|   | c1de162c99 | ||
|   | a7ef8aba68 | ||
|   | 3ee4c11a99 | ||
|   | 990ae10dc2 | ||
|   | 52b2fd046b | ||
|   | 9f41f80a91 | ||
|   | eec4a91ad8 | ||
|   | 7c51001c3c | ||
|   | a4ea4b1f5f | ||
|   | 19fc37539e | ||
|   | ce7acb0feb | ||
|   | 105b7678b8 | ||
|   | b67575586e | ||
|   | 3dc6898673 | ||
|   | a73754c1b5 | ||
|   | 1ebf1c00d6 | ||
|   | 7dac7d757e | ||
|   | b1f3192b95 | ||
|   | 16984d18bb | ||
|   | e603893d77 | ||
|   | a7998b30c6 | ||
|   | 3277a4e8c3 | ||
|   | 7e769d0e14 | ||
|   | 713e0579f8 | ||
|   | 6e130cc020 | ||
|   | eb036a12d9 | ||
|   | 534d1f5055 | ||
|   | cbef909657 | ||
|   | 874f3b32b3 | ||
|   | 2fd017cf73 | ||
|   | 5740b018a7 | ||
|   | 288bf6805a | ||
|   | 02d37a369a | ||
|   | 1d316c3258 | ||
|   | a56ce62f1a | ||
|   | c268f42851 | ||
|   | 7251e802ab | ||
|   | 5b1a2d10c2 | ||
|   | 2dd7f292b1 | ||
|   | 213c53e307 | ||
|   | ce07dfd8ac | ||
|   | c1dba462e8 | ||
|   | 47f0d74812 | ||
|   | ce80285f8d | ||
|   | d2dd1a43dd | ||
|   | 12d73fe90d | ||
|   | c2741638b2 | ||
|   | 4a7fb3d509 | ||
|   | f6ff652ca4 | ||
|   | 6165cb0f83 | ||
|   | 1f361b7b10 | ||
|   | 5269ff978b | ||
|   | 55595493a9 | ||
|   | ad3ff0aba7 | ||
|   | ce48546cef | ||
|   | 35b3bc995e | ||
|   | 63f60019d1 | ||
|   | 0d741b6275 | ||
|   | 0df9080bbb | ||
|   | ddcf89e6a2 | ||
|   | 5de225d5d4 | ||
|   | 5cddb482f1 | ||
|   | c000d724de | ||
|   | 504055f331 | ||
|   | 7f6880f40e | ||
|   | 02e4e3c892 | ||
|   | 993d73c359 | ||
|   | 97ca0b818e | ||
|   | 44166f76d4 | ||
|   | 557d6d37a1 | ||
|   | d3ad56a307 | ||
|   | 0641022ec5 | ||
|   | 80c7a8473a | ||
|   | d9a954ca91 | ||
|   | c219f64322 | ||
|   | f7a9ecff21 | ||
|   | 2b3126ae04 | ||
|   | 934c227545 | ||
|   | cc0515c217 | ||
|   | 55ba75f2bc | ||
|   | c220228566 | ||
|   | 26b476ab3c | ||
|   | b8a67d530f | ||
|   | b08c96d2db | ||
|   | 4773c39a57 | ||
|   | 892843b290 | ||
|   | 733244531e | ||
|   | 66633273e2 | ||
|   | 0405adcd16 | ||
|   | 426a7ac8dd | ||
|   | 3bf6205ff7 | ||
|   | c7f4986e61 | ||
|   | 0f0a3fdaf7 | ||
|   | 7d6911b140 | ||
|   | b8777539d7 | ||
|   | b5b1849ab3 | ||
|   | 0e10c81025 | ||
|   | 5fc0eaef1a | ||
|   | 113718c3c1 | ||
|   | 701bea6cae | ||
|   | 8d516ed12a | ||
|   | 667c5744f2 | ||
|   | 80b7c840e2 | ||
|   | 919c86796f | ||
|   | c90c88ecbf | ||
|   | d9ba0e2c46 | ||
|   | 45b2fc590b | ||
|   | 17ffdb0247 | ||
|   | c2fba15fc6 | ||
|   | 5937be695f | ||
|   | a076fcde84 | ||
|   | ede9931903 | ||
|   | 722e01608c | ||
|   | af926370d6 | ||
|   | 5971aee02e | ||
|   | cce7ad449a | ||
|   | d437dd5919 | ||
|   | f1980730d2 | ||
|   | 47773e9cae | ||
|   | 60969b0916 | ||
|   | ecc7925d03 | ||
|   | 6d3010dcc7 | ||
|   | 0164bafbf1 | ||
|   | 3940606167 | ||
|   | da9faccada | ||
|   | 7e708b3bf7 | ||
|   | 05630c9896 | ||
|   | 369c56db73 | ||
|   | 9873459169 | ||
|   | 7776b3766b | ||
|   | 29c9004654 | ||
|   | 601c909004 | ||
|   | 72aa9a3b62 | ||
|   | 2ecf7bca97 | ||
|   | cbdfaccdb2 | ||
|   | 93d1b9a2d5 | ||
|   | bfb5ee794e | ||
|   | 9ae8bd238b | ||
|   | 98a64e3114 | ||
|   | 0171f3aec7 | ||
|   | 2c827bab9a | ||
|   | ec920093d4 | ||
|   | 4289ff6652 | ||
|   | cd32ef60da | ||
|   | 6ef3d091e1 | ||
|   | 0f59496778 | ||
|   | 795b670b3c | ||
|   | 4ad65e3d69 | ||
|   | f48b1150cf | ||
|   | 198db8bc68 | ||
|   | 3ae447ca44 | ||
|   | a2ec878ef0 | ||
|   | d43b5d3337 | ||
|   | 11e3503dc2 | ||
|   | accf44b769 | ||
|   | b612c0e0d6 | ||
|   | 771c7518e6 | ||
|   | 4f2bad034a | ||
|   | 0456669aeb | ||
|   | 2eb71321e7 | ||
|   | d2e9e22e4e | ||
|   | e2427c8dce | ||
|   | 26162815c8 | ||
|   | d0c86ea2f7 | ||
|   | 4f5b4d4472 | ||
|   | 10d71587e7 | ||
|   | 9ac777d687 | ||
|   | 7d1f9f3981 | ||
|   | a214ab463e | ||
|   | 2f21f6ef8a | ||
|   | 590cd8500d | ||
|   | f48a28264f | ||
|   | 51953bce09 | ||
|   | c445d4b839 | ||
|   | 9dabce1dd7 | ||
|   | 6b77f08d86 | ||
|   | 50e03d41ab | ||
|   | 4759c89628 | ||
|   | 458ab9b1e0 | ||
|   | dfb2a7153b | ||
|   | e9141d82f3 | ||
|   | d9f42712e4 | ||
|   | da0658f3d9 | ||
|   | 412a5a6d8e | ||
|   | 9cf9e6edd8 | ||
|   | 28050fc9fb | ||
|   | ea9f227fa8 | ||
|   | 05164e90fb | ||
|   | 97156ccf8a | ||
|   | 003d55968e | ||
|   | 98dd21b75c | ||
|   | a349e34bc2 | ||
|   | 245c825cbf | ||
|   | 4ba6698c4b | ||
|   | dac6edea90 | ||
|   | b2e2dbe0c2 | ||
|   | f822fa0d57 | ||
|   | bbee94edba | ||
|   | c39c408332 | ||
|   | dde4a40e69 | ||
|   | 1afa4d2868 | ||
|   | d25f371d6c | ||
|   | 948261d1d1 | ||
|   | 04b7749f85 | ||
|   | 8668f0ff5b | ||
|   | b7d8ff6c99 | ||
|   | 0894a30f13 | ||
|   | bd511887a7 | ||
|   | 0304c0eca0 | ||
|   | 6d5c6e2fbc | ||
|   | e7e7912519 | ||
|   | b9c41d8f99 | ||
|   | 25a2264ac3 | ||
|   | ad64657c74 | ||
|   | 106f18a73a | ||
|   | 0fb38140a6 | ||
|   | 60133941ae | ||
|   | 76db64c073 | ||
|   | 24b390ba03 | ||
|   | b4d34d9085 | ||
|   | f80bea27a9 | ||
|   | 4b846964be | ||
|   | e6357e2eef | ||
|   | d538b6145f | ||
|   | 229ba3a75c | ||
|   | 672b867847 | ||
|   | ca1dacd621 | ||
|   | 26abb4301d | ||
|   | 045dac59af | ||
|   | 1ffa87ea23 | ||
|   | 357df2bf68 | ||
|   | 24f3db3e3f | ||
|   | 4bb51b0120 | ||
|   | 32b9c1eba1 | ||
|   | 40bffc1d25 | ||
|   | ed2c4e2642 | ||
|   | b3353e220a | ||
|   | 7dc917babf | ||
|   | 00dbba04a2 | ||
|   | 3efd1a0451 | ||
|   | b7d7ca4014 | ||
|   | 854a54e9c6 | ||
|   | 4f4edb109f | ||
|   | 265bfeb889 | ||
|   | 96110637d9 | ||
|   | ad34f98e6d | ||
|   | a8a1563586 | ||
|   | 9b25a54a47 | ||
|   | 4b8c96c769 | ||
|   | c62ff85e73 | ||
|   | 7d5a27ec0f | ||
|   | d6aba040dd | ||
|   | ca4757db5b | ||
|   | c917b67cbd | ||
|   | 9659c97978 | ||
|   | 7d862d6f2a | ||
|   | 9c80776d8c | ||
|   | d5cd288fe8 | ||
|   | 239e817779 | ||
|   | 1986215919 | ||
|   | 239f5f1a2f | ||
|   | 3bca32c6d5 | ||
|   | 183eff745d | ||
|   | 4392d78ff6 | ||
|   | 858196ab53 | ||
|   | fb75d8c1f2 | ||
|   | 7628569579 | ||
|   | 8a9d5f7753 | ||
|   | cdcccf5089 | ||
|   | de95c92e2d | ||
|   | 3030b8d476 | ||
|   | 92ed14c0e4 | ||
|   | 5b94a4de9a | ||
|   | 709112c498 | ||
|   | e465ec8835 | ||
|   | f6eb31bf9d | ||
|   | 426f939982 | ||
|   | fab6cebf0d | ||
|   | ff081dd0f1 | ||
|   | 868399ed6f | ||
|   | 1bc9b95289 | ||
|   | 9af805ab5e | ||
|   | 6b88081360 | ||
|   | 667c828359 | ||
|   | 50d37ce4f6 | ||
|   | af0246cd27 | ||
|   | 857e4e49d8 | ||
|   | c1afed7f98 | ||
|   | 5480e54185 | ||
|   | 99d0a0a6fd | ||
|   | 8a998369d6 | ||
|   | 8b490c5047 | ||
|   | 7e70ba6ab2 | ||
|   | 90e09fc384 | ||
|   | 266f2e763d | ||
|   | c979cfb912 | ||
|   | 8ee29b1e43 | ||
|   | 26fbc07cac | ||
|   | f01fe65be4 | ||
|   | 3fdd6a80f9 | ||
|   | da1de8db1d | ||
|   | dd1bf7b49d | ||
|   | f18913b5a0 | ||
|   | a2cd227f1a | ||
|   | 78e64e1f60 | ||
|   | 23a9b79320 | ||
|   | 76394ce341 | ||
|   | 1935df1faa | ||
|   | 5af4ce28ab | ||
|   | ce8ee569c4 | ||
|   | b0508f430e | ||
|   | 2139a80a7a | ||
|   | aa4bc2ce03 | ||
|   | fa65f84e09 | ||
|   | c06357a351 | ||
|   | 092a02a624 | ||
|   | b9699f745f | ||
|   | 7fa9f10c30 | ||
|   | 7bf0655dae | ||
|   | 96c5fdcbeb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c2e6d40382 | ||
|   | 810d2a1ceb | ||
|   | af74f21af9 | ||
|   | cdf7558a8e | ||
|   | 41b86e6c10 | ||
|   | 085c6f8bdd | ||
|   | 3039c678a5 | ||
|   | 498882d014 | ||
|   | 6c2b8c2abb | ||
|   | e955cc4378 | ||
|   | eb96dd4803 | ||
|   | e0bdef98a6 | ||
|   | 1130007d14 | ||
|   | d99d092784 | ||
|   | e3b18a33ca | ||
|   | 1890aab1e6 | ||
|   | 42bf350034 | ||
|   | 5ff52ea113 | ||
|   | 432e3ba636 | ||
|   | f7ab52fe9a | ||
|   | ad8430049d | ||
|   | 2dffe7ba9e | ||
|   | 5b8f97e0f6 | ||
|   | b3a763a48d | ||
|   | 07569f10b5 | ||
|   | 7c5a78a1cf | ||
|   | 5cca5bfe86 | ||
|   | 0e021e7d7d | ||
|   | b30ee884a7 | ||
|   | 869b7c85ca | ||
|   | 4d0d1ed2a1 | ||
|   | 291983e4c3 | ||
|   | 909cff2158 | ||
|   | 4e676b1dba | ||
|   | 9149bb9333 | ||
|   | 4631994f20 | ||
|   | 82e9178320 | ||
|   | 67b4688168 | ||
|   | 6e0e169b6e | ||
|   | 100ba8edfa | ||
|   | d7448ecb95 | ||
|   | 8b1801f378 | ||
|   | 01a4d57566 | ||
|   | 7edc9064d9 | ||
|   | 30c47a65f4 | ||
|   | 0889f42a00 | ||
|   | 4999f1ad51 | ||
|   | f15fbe53cf | ||
|   | 046f7b5153 | ||
|   | 5339fe6e06 | ||
|   | de7ffb10cb | ||
|   | 80224e6974 | ||
|   | 0c7c536f73 | ||
|   | e5c386c39a | ||
|   | bb2462483e | ||
|   | d5bc498373 | ||
|   | 979b7ae651 | ||
|   | c73330a466 | ||
|   | efe8eca4e3 | ||
|   | a37aad18b7 | ||
|   | cfa0c45213 | ||
|   | 509481ef06 | ||
|   | 9aa8175e23 | ||
|   | 76f59d99a2 | ||
|   | bd66bd6cf0 | ||
|   | d69333dea4 | ||
|   | 3fd7899b93 | ||
|   | 8f8a2cea56 | ||
|   | 879011c8e9 | ||
|   | d5794c3e2e | ||
|   | 61dbae8b8b | ||
|   | fcc22ba560 | ||
|   | 2adeb88fe6 | ||
|   | e63a78bcdb | ||
|   | b065f002a4 | ||
|   | 349a5f52b1 | ||
|   | aa5e20df05 | ||
|   | 793b9f238c | ||
|   | 9c4fdaa4f3 | ||
|   | d1a9cb488a | ||
|   | faee2c3e1b | ||
|   | b7845c318e | ||
|   | 426a0727c3 | ||
|   | 584e509a9c | ||
|   | f3639c2663 | ||
|   | 1431e75f8b | ||
|   | be8812e0af | ||
|   | fd6436d490 | ||
|   | fd1342f9d1 | ||
|   | 5fa0012195 | ||
|   | 9dbb67ef01 | ||
|   | d16e2f37d4 | ||
|   | d9e8b53ffe | ||
|   | 1997e63b7c | ||
|   | 6f673359ff | ||
|   | 45dfbff10a | ||
|   | 348ee96274 | ||
|   | 8edee32e77 | ||
|   | 6d8d263ca6 | ||
|   | 35923709e2 | ||
|   | fdd4d53448 | ||
|   | 06419f662e | ||
|   | 57763ef032 | ||
|   | 8e506f7749 | ||
|   | c7f8fe1468 | ||
|   | 4156a4e36d | ||
|   | ba3cc7df0f | ||
|   | 0c212d39eb | ||
|   | 3bd2e8dbf5 | ||
|   | 5292119e6e | ||
|   | 994a397231 | ||
|   | 353b71f803 | ||
|   | eb12afe8cc | ||
|   | 4a176f1b43 | ||
|   | 8e228baa82 | ||
|   | 154b53b0d8 | ||
|   | a3f680d80c | ||
|   | 0d75fe6b81 | ||
|   | 4070380ded | ||
|   | 41195dcef0 | ||
|   | 78a1e45be2 | ||
|   | d8e88bc58d | ||
|   | 448e9b71b8 | ||
|   | 2e178164cc | ||
|   | 9f2e3f05a1 | ||
|   | 405bd29ebd | ||
|   | b39b54e0ac | ||
|   | 119c5c9071 | ||
|   | 7a4c9b128c | ||
|   | dc5b92030f | ||
|   | db0a010d7c | ||
|   | a117a19bdf | ||
|   | 5f46fdb406 | ||
|   | f0201de4cc | ||
|   | 6cd51a318b | ||
|   | c1a4b27bc7 | ||
|   | 5113222050 | ||
|   | 90f12eea5e | ||
|   | 2403743701 | ||
|   | 3e6a759309 | ||
|   | 35a430e9f4 | ||
|   | b644f7d23d | ||
|   | 7702a05464 | ||
|   | 493af5fe82 | ||
|   | ac66a59cec | ||
|   | e10c8faa47 | ||
|   | 9b7d17433c | ||
|   | a40eb1ff43 | ||
|   | 04df6c3e9e | ||
|   | 1b970e5a66 | ||
|   | 75406c2d01 | ||
|   | 64d3511fbc | ||
|   | c610f54977 | ||
|   | 090ad34f78 | ||
|   | 358c5205d2 | ||
|   | 5503cd0589 | ||
|   | dae42b1bd9 | ||
|   | 06a25284e8 | ||
|   | 5989560f15 | ||
|   | 63c995e5da | ||
|   | dc5607f554 | ||
|   | d49302c032 | ||
|   | 63fef9bd4b | ||
|   | 6599351d45 | ||
|   | 47e9531972 | ||
|   | 3ba31483f4 | ||
|   | f4ca94f2e1 | ||
|   | 67f9be2b77 | ||
|   | e2fd155e1b | ||
|   | 931068dede | ||
|   | bc4c9cc40d | ||
|   | 294665fbe8 | ||
|   | e8f6a79c8f | ||
|   | 5fd8b5c5b9 | ||
|   | 226b2a73af | ||
|   | 42d421a6fc | ||
|   | a90203f256 | ||
|   | c3ef79caa9 | ||
|   | 1439afcd9c | ||
|   | d263b19910 | ||
|   | 1e477226ea | ||
|   | 026fc1d2e3 | ||
|   | 2d4bd9857a | ||
|   | 8f48f5b45c | ||
|   | 22210b7400 | ||
|   | 7d05855ee0 | ||
|   | b2460cbc3d | ||
|   | 4561957e56 | ||
|   | 3367fadc3a | ||
|   | d7e409b042 | ||
|   | a0b28e8ad1 | ||
|   | f928a8e58e | ||
|   | 0bc4b3d0fa | ||
|   | e352768388 | ||
|   | 6835b73e49 | ||
|   | f1503f871b | ||
|   | c4d8aba5c8 | ||
|   | 39f24c41ad | ||
|   | 21644ec889 | ||
|   | 613470b44d | ||
|   | 6c918e346b | ||
|   | bce8539572 | ||
|   | aab86e00ec | ||
|   | 2a58726caf | ||
|   | 4163b35b32 | ||
|   | 9c6dac8180 | ||
|   | 80fc37724b | ||
|   | 77b25f5132 | ||
|   | 684f098450 | ||
|   | d09f74d30f | ||
|   | 3d973b112e | ||
|   | 96986164a4 | ||
|   | 78152c20a9 | ||
|   | 2bb64e9e2f | ||
|   | 746844dfc8 | ||
|   | 41b613a2d7 | ||
|   | 3b3aeea224 | ||
|   | 71c592a0ce | ||
|   | 15193fcf5f | ||
|   | a31f53395f | ||
|   | 283b134d84 | ||
|   | 271eb614cd | ||
|   | 16167bef07 | ||
|   | 1eac9fa1cd | ||
|   | 7f819f0020 | ||
|   | dec1f99a5f | ||
|   | c705e74fc8 | ||
|   | 01df10f93e | ||
|   | 9877f08cf4 | ||
|   | 3dc4b1d775 | ||
|   | 02791c51ae | ||
|   | 49683326e6 | ||
|   | 947773a82e | ||
|   | 2a229df624 | ||
|   | e605ad5e46 | ||
|   | 0d4f43472b | ||
|   | b30e467685 | ||
|   | a56c0b52d5 | ||
|   | c17ebfd279 | ||
|   | 5400b1da96 | ||
|   | 69f4a618b2 | ||
|   | 16b8b6698c | ||
|   | b29a700d40 | ||
|   | bbb1468439 | ||
|   | 72f9d6a8d3 | ||
|   | 3ec8da1f17 | ||
|   | dbea3848df | ||
|   | 33871435e1 | ||
|   | f1f22b43dc | ||
|   | 2fb9a56e0b | ||
|   | 14e8f66ed7 | ||
|   | e6f5072462 | ||
|   | a64f50fa72 | ||
|   | bb5f6e88d0 | ||
|   | 6991403203 | ||
|   | 410bd22f8a | ||
|   | b81d823602 | ||
|   | bd5115f9aa | ||
|   | 7bcbed80d7 | ||
|   | 8fb62ebf5f | ||
|   | 209dd9923f | ||
|   | c75207e391 | ||
|   | d957f36927 | ||
|   | 9ac459b6d9 | ||
|   | e08b2817ba | ||
|   | 4ca13c409b | ||
|   | 0d515e2303 | ||
|   | a2153bc6aa | ||
|   | ca171afe6f | ||
|   | bf4e97bd48 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8c59a12a03 | ||
|   | 89569355be | ||
|   | 3a41b3bdcf | ||
|   | 12bd7037b3 | ||
|   | ca4f573be0 | ||
|   | 07fceeab5a | ||
|   | 3aa376e912 | ||
|   | 92d30a8896 | ||
|   | 83876fb9da | ||
|   | 29bdf7877c | ||
|   | 29199e2782 | ||
|   | 68e1378615 | ||
|   | cf7efb5bfc | ||
|   | 8634ee536d | ||
|   | 632d3cda24 | ||
|   | 29b6a907d4 | ||
|   | 7474d09e5d | ||
|   | fc7bcd7e00 | ||
|   | f6fb2e4b1d | ||
|   | 8c8673a272 | ||
|   | 4404a1173b | ||
|   | e08c10315e | ||
|   | 16473c9177 | ||
|   | 235fd5603f | ||
|   | d07d5832f5 | ||
|   | ef8be5d559 | ||
|   | ccafdc6e1f | ||
|   | 11827aa4c0 | ||
|   | 6b0589d343 | ||
|   | cec1eed99e | ||
|   | d7e1e9e284 | ||
|   | cae46453a7 | ||
|   | a6e948c808 | ||
|   | 7638020bfc | ||
|   | 10a62ca17c | ||
|   | 0afc7c184f | ||
|   | 168e26aeb4 | ||
|   | e6b9389b33 | ||
|   | 377c37425e | ||
|   | 4af26602bb | ||
|   | c6624e5cb6 | ||
|   | f7ae5b91bf | ||
|   | 07e68496c0 | ||
|   | d5a947e2cc | ||
|   | 3f920767f1 | ||
|   | 3e14d27a1e | ||
|   | cfa4c14108 | ||
|   | 209056dbe1 | ||
|   | 10356a7496 | ||
|   | d4ae74de44 | ||
|   | 88d5e7dd5e | ||
|   | 06c7b0b82e | ||
|   | 689febda60 | ||
|   | 80bc6fda8b | ||
|   | 346eb78c4e | ||
|   | 2df02f1b09 | ||
|   | 92915eddc2 | ||
|   | cddbf460f8 | ||
|   | 3c63c23e5a | ||
|   | ba67b1291f | ||
|   | 7bced28327 | ||
|   | db2b60700c | ||
|   | 9034822c44 | ||
|   | e8254f9aae | ||
|   | a14179b81a | ||
|   | 427c5db7f4 | ||
|   | fcb5865468 | ||
|   | 41370be2b8 | ||
|   | d7d8dd8986 | ||
|   | a0f596e419 | ||
|   | 0a8894feb7 | ||
|   | 1db9eea0f8 | ||
|   | 489783c398 | ||
|   | be62f327ee | ||
|   | 32359adb6d | ||
|   | 2e198af8c3 | ||
|   | d154fcbd71 | ||
|   | 21e277b8a2 | ||
|   | f98cdd0749 | ||
|   | e60e306426 | ||
|   | ec36d396d9 | ||
|   | 135232d880 | ||
|   | 9c42ca0315 | ||
|   | 9ad9c569a6 | ||
|   | a9071d7920 | ||
|   | 1b4a10fac1 | ||
|   | d340f3b383 | ||
|   | f8c5eeab5d | ||
|   | 9cd2d0df93 | ||
|   | 78914091b1 | ||
|   | e12c324613 | ||
|   | 7cf396b518 | ||
|   | 8b3b40e627 | ||
|   | 90e14762e3 | ||
|   | d1dd8231cd | ||
|   | e70a3e09bf | ||
|   | 98656b0044 | ||
|   | a48aa3c778 | ||
|   | 05d7b98ba0 | ||
|   | f291ea6647 | ||
|   | 5d6e332044 | ||
|   | acb471fbe5 | ||
|   | 894f4379e6 | ||
|   | 1c73007ae6 | ||
|   | 2f7d744228 | ||
|   | e2cba90f8d | ||
|   | 352214ba0a | ||
|   | bd9b72fb22 | ||
|   | 50c9a667b3 | ||
|   | 3d32e6310d | ||
|   | 3bc54aa9e0 | ||
|   | def1ec3518 | ||
|   | 077802f972 | ||
|   | 914b47f340 | ||
|   | b2a78fd063 | ||
|   | 7d1835e59c | ||
|   | 833ccf3637 | ||
|   | 51be916f39 | ||
|   | e375408777 | ||
|   | 5078dc1cbf | ||
|   | 875148366e | ||
|   | c9ec4b4e24 | ||
|   | efa2b2db27 | ||
|   | 8ce120b74d | ||
|   | 26e678a97d | ||
|   | e71dd7409e | ||
|   | 58ffc2c6ca | ||
|   | d3f29362b9 | ||
|   | b429fe8254 | ||
|   | e1cb549b28 | ||
|   | 65a22257cc | ||
|   | e2f753eaa7 | ||
|   | c7127b65bf | ||
|   | 0c58c3572a | ||
|   | 26ae5fd728 | ||
|   | 370d92213b | ||
|   | 6e8321a22a | ||
|   | a8e8c1ce5d | ||
|   | a8a8cafd2b | ||
|   | b609890f28 | ||
|   | aac09ae092 | ||
|   | f1ff872944 | ||
|   | b195d2980a | ||
|   | d11736181f | ||
|   | 3e84486dd5 | ||
|   | a674ce36e4 | ||
|   | f6569a2625 | ||
|   | da10da79b3 | ||
|   | f236b76d5c | ||
|   | a71c22bedd | ||
|   | cc528e41cf | ||
|   | 351962475f | ||
|   | 6c73392a57 | ||
|   | 072ad87831 | ||
|   | 370a1f0574 | ||
|   | 9ca7aca4b7 | ||
|   | 0f2e9f66b1 | ||
|   | d03c3ab713 | ||
|   | 57c0b34ae9 | ||
|   | 06a94f0f28 | ||
|   | 750e7b1262 | ||
|   | 19e32752bb | ||
|   | 89c0729964 | ||
|   | fd07152aea | ||
|   | b656f189b6 | ||
|   | 9ac8d70152 | ||
|   | 8cc0b46335 | ||
|   | 1f15094da7 | ||
|   | b881adb853 | ||
|   | 4bfc3a5629 | ||
|   | ae6c0bfe40 | ||
|   | 4ce9c71521 | ||
|   | ec48323a7d | ||
|   | 7d9bae16cd | ||
|   | 163ff3d4e4 | ||
|   | 43fbf97e10 | ||
|   | 71faaf2ab1 | ||
|   | 7b0e743eca | ||
|   | 31a0c53855 | ||
|   | e8996063dd | ||
|   | 00842a3354 | ||
|   | d33f18ecb7 | ||
|   | fb7f620316 | ||
|   | 712e0d3e3b | ||
|   | 1870dc29c0 | ||
|   | da6fdc74d8 | ||
|   | 515e39154a | ||
|   | ff7731d063 | ||
|   | e9a3666dd5 | ||
|   | 55c56d53f4 | ||
|   | e4d55e6842 | ||
|   | d8661cf2db | ||
|   | 8815b126b5 | ||
|   | 2cd367f29f | ||
|   | 7395d19489 | ||
|   | d55cb95479 | ||
|   | 68ece7d363 | ||
|   | 6e4a8ac6df | ||
|   | 790629849f | ||
|   | f0443a43b2 | ||
|   | 3041eb5ce0 | ||
|   | c69247f190 | ||
|   | 27d6a62e67 | ||
|   | 6e7fc914aa | ||
|   | 65d587843b | ||
|   | c54792af22 | ||
|   | 7b7e023103 | ||
|   | 7637d36146 | ||
|   | 6c62afb123 | ||
|   | 44210ce6f2 | ||
|   | e21efc0a5c | ||
|   | 09a965022f | ||
|   | 7534ecd2f2 | ||
|   | 46bf5cf830 | ||
|   | 7ba7761a57 | ||
|   | 5268afabdb | ||
|   | 3ea7506003 | ||
|   | ee14d206c8 | ||
|   | b65f4b9af6 | ||
|   | 6d000a3f9a | ||
|   | 9ba0de67f5 | ||
|   | d22eaa1318 | ||
|   | 3f4bfab7fe | ||
|   | 9292f217c5 | ||
|   | 3b779bf423 | ||
|   | ea410d3af1 | ||
|   | 4e71c2c500 | ||
|   | 454ddf366a | ||
|   | d0ba5696d1 | ||
|   | c53fd0d1e1 | ||
|   | 7bbecfde2b | ||
|   | a06f378582 | ||
|   | b3b42b741d | ||
|   | 020f115d7c | ||
|   | 2cc9d70915 | ||
|   | b242c6651a | ||
|   | 14a51799a6 | ||
|   | 79a6dacd2f | ||
|   | 6891f1df1c | ||
|   | 497494620d | ||
|   | 7a13242077 | ||
|   | b9d6973a79 | ||
|   | ed0e8c5e8d | ||
|   | d8f530f8ac | ||
|   | a46874b7ff | ||
|   | cf68f25a03 | ||
|   | a496563b5c | ||
|   | a763ad5bf1 | ||
|   | f0b0200932 | ||
|   | 342f22e6a1 | ||
|   | 372ecc6557 | ||
|   | 16c604937e | ||
|   | 5cbffb23fd | ||
|   | 6de38d3b85 | ||
|   | b34ce577d9 | ||
|   | 4b9fcd7de7 | ||
|   | cc71ccaafa | ||
|   | 9ff2eece3a | ||
|   | 2bc97cc9c8 | ||
|   | a87570cf5a | ||
|   | 9ffd25e3a0 | ||
|   | 61fdab294a | ||
|   | d4e137bb58 | ||
|   | c251e4f241 | ||
|   | a55d0f347b | ||
|   | f15cc0b424 | ||
|   | 4e17875011 | ||
|   | 256b64b6b3 | ||
|   | 8c0c0592e2 | ||
|   | f53f81dbc4 | ||
|   | d6c85719c9 | ||
|   | c51c80bf47 | ||
|   | 544832756d | ||
|   | ca8586789a | ||
|   | 1afc2b3518 | ||
|   | 17352ea5bd | ||
|   | 6f5e3c2711 | ||
|   | bee21cd3fe | ||
|   | c4340e05d2 | ||
|   | 6d6eef4e97 | ||
|   | 71137032df | ||
|   | ffff4dc03d | ||
|   | 4033131f2e | ||
|   | 65b16c763e | ||
|   | 33af3de4a3 | ||
|   | a822c1eb2f | ||
|   | 4eb46bc275 | ||
|   | ccc9b73f9b | ||
|   | e9ffdeff19 | ||
|   | a0ab4dffc9 | ||
|   | f5f8ad0e02 | ||
|   | 82957ff6ef | ||
|   | 0864aeb9c6 | ||
|   | cda6310373 | ||
|   | 26a87e9280 | ||
|   | 0b16a4880a | ||
|   | db68c5852c | ||
|   | 256aec5308 | ||
|   | 20dd3ca21c | ||
|   | 25cc76e022 | ||
|   | 168cc607aa | ||
|   | edc4601f8e | ||
|   | 8f86a7ad8c | ||
|   | 4d7ad0dc51 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 34ac80567e | ||
|   | 23bdc03f29 | ||
|   | 3099748a6d | ||
|   | e384f76ac1 | ||
|   | 7050d19be7 | ||
|   | ca678330d3 | ||
|   | 10a5b3f9c3 | ||
|   | 0fcedc5046 | ||
|   | f558f7fb8c | ||
|   | 06207defe7 | ||
|   | 986f9d7633 | ||
|   | 81277fd12e | ||
|   | 30442b25c0 | ||
|   | 67ac3b4ba3 | ||
|   | f819e2cf8d | ||
|   | a376f4525b | ||
|   | 5c1553286a | ||
|   | 2c5d3f7492 | ||
|   | 58f01ba11a | ||
|   | 3fdf9a2e28 | ||
|   | 4cbd8e7673 | ||
|   | 20ca642e51 | ||
|   | 05ad7ea011 | ||
|   | faea8c9f4c | ||
|   | 0d4c51f26e | ||
|   | 2c1b25b00b | ||
|   | 407f305d21 | ||
|   | 5d5d6b247f | ||
|   | 77bd7c37c1 | ||
|   | 783ea0f5c8 | ||
|   | 33f1b45f30 | ||
|   | e1df50ad3b | ||
|   | 6a9a4cf65f | ||
|   | 443634ecf0 | ||
|   | 2a17870d6d | ||
|   | 486ed7dcaa | ||
|   | 19c13096dc | ||
|   | 49b4271a47 | ||
|   | 404d0b8d05 | ||
|   | fe63c12cd9 | ||
|   | 44023c3db7 | ||
|   | 6242997849 | ||
|   | ec5d7508c9 | ||
|   | 7461d8f806 | ||
|   | 362236d7df | ||
|   | ba384b9eee | ||
|   | d94df728e5 | ||
|   | d0a53d1760 | ||
|   | 2e5ec1f0c1 | ||
|   | 6c8aedfb8b | ||
|   | f354e1eb0f | ||
|   | 49b1e5897e | ||
|   | 1a58c17180 | ||
|   | 7fb852893c | ||
|   | aa9a354746 | ||
|   | 78f5429c92 | ||
|   | 79de8e0f32 | ||
|   | 70f59eeec6 | ||
|   | b75792a3bf | ||
|   | 5cb7117656 | ||
|   | 3a5bd7474b | ||
|   | 304fad3f49 | ||
|   | fb929a089c | ||
|   | 6076a0cdc4 | ||
|   | a3736683eb | ||
|   | 413f1c31bb | ||
|   | 20baff380b | ||
|   | 582d159884 | ||
|   | 9c43c5806d | ||
|   | 8ad2bf5401 | ||
|   | 66409f0fa5 | ||
|   | 68172e006f | ||
|   | cf5e808a96 | ||
|   | 3faebaeb4b | ||
|   | 6b8cfe661c | ||
|   | 2fd64af737 | ||
|   | 050cdf3783 | ||
|   | d73b3d77ea | ||
|   | acc024bcf9 | ||
|   | deb179ad38 | ||
|   | bfb11790a2 | ||
|   | af23110074 | ||
|   | b8e71609db | ||
|   | 1876b3827f | ||
|   | 38d3b8d087 | ||
|   | c5ef33cc78 | ||
|   | 7427b209a7 | ||
|   | 71397e5199 | ||
|   | 1bc3b3befc | ||
|   | 872e46a076 | ||
|   | ad386c0e22 | ||
|   | 7e281f66c2 | ||
|   | 7daafcbe1b | ||
|   | c5b223988a | ||
|   | 6cc9ce573b | ||
|   | 23192226dd | ||
|   | 736444201b | ||
|   | 785f49b005 | ||
|   | 55dd1f4aa1 | ||
|   | 6d0490d7d9 | ||
|   | 06667455ae | ||
|   | 3640960486 | ||
|   | 4ad3dbf3e2 | ||
|   | 10957deb1f | ||
|   | 0a128db269 | ||
|   | 6bb3b84377 | ||
|   | b8d2c551e0 | ||
|   | 34e06351fb | ||
|   | e179404a9e | ||
|   | 5f81a204f2 | ||
|   | df3b70a533 | ||
|   | e4607735ff | ||
|   | 389b7def0b | ||
|   | e35bd30ed3 | ||
|   | 5cc4e2bb16 | ||
|   | 3a453f5843 | ||
|   | 28e0384b55 | ||
|   | dcd6c6f06f | ||
|   | 28d26065e4 | ||
|   | 300c8d06c4 | ||
|   | d2a1d11d16 | ||
|   | 6ae717bbfe | ||
|   | 21296b4224 | 
							
								
								
									
										13
									
								
								.devcontainer/Dockerfile
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,13 @@ | |||||||
|  | # See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile | ||||||
|  | FROM mcr.microsoft.com/vscode/devcontainers/python:0-3.9 | ||||||
|  |  | ||||||
|  | ENV \ | ||||||
|  |   DEBIAN_FRONTEND=noninteractive \ | ||||||
|  |   DEVCONTAINER=true \ | ||||||
|  |   PATH=$PATH:./node_modules/.bin | ||||||
|  |  | ||||||
|  | # Install nvm | ||||||
|  | COPY .nvmrc /tmp/.nvmrc | ||||||
|  | RUN \ | ||||||
|  |   su vscode -c \ | ||||||
|  |     "source /usr/local/share/nvm/nvm.sh && nvm install $(cat /tmp/.nvmrc) 2>&1" | ||||||
							
								
								
									
										31
									
								
								.devcontainer/devcontainer.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,31 @@ | |||||||
|  | { | ||||||
|  |   "name": "Home Assistant Frontend", | ||||||
|  |   "build": { | ||||||
|  |     "dockerfile": "Dockerfile", | ||||||
|  |     "context": ".." | ||||||
|  |   }, | ||||||
|  |   "appPort": 8123, | ||||||
|  |   "context": "..", | ||||||
|  |   "postCreateCommand": "script/bootstrap", | ||||||
|  |   "extensions": [ | ||||||
|  |     "github.vscode-pull-request-github", | ||||||
|  |     "dbaeumer.vscode-eslint", | ||||||
|  |     "ms-vscode.vscode-typescript-tslint-plugin", | ||||||
|  |     "esbenp.prettier-vscode", | ||||||
|  |     "bierner.lit-html", | ||||||
|  |     "runem.lit-plugin", | ||||||
|  |     "ms-python.vscode-pylance" | ||||||
|  |   ], | ||||||
|  |   "settings": { | ||||||
|  |     "terminal.integrated.shell.linux": "/bin/bash", | ||||||
|  |     "files.eol": "\n", | ||||||
|  |     "editor.tabSize": 2, | ||||||
|  |     "editor.formatOnPaste": false, | ||||||
|  |     "editor.formatOnSave": true, | ||||||
|  |     "editor.formatOnType": true, | ||||||
|  |     "[typescript]": { | ||||||
|  |       "editor.defaultFormatter": "esbenp.prettier-vscode" | ||||||
|  |     }, | ||||||
|  |     "files.trimTrailingWhitespace": true | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,4 +0,0 @@ | |||||||
| node_modules |  | ||||||
| hass_frontend |  | ||||||
| hass_frontend_es5 |  | ||||||
| .git |  | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| { | { | ||||||
|   "extends": [ |   "extends": [ | ||||||
|     "plugin:@typescript-eslint/recommended", |  | ||||||
|     "airbnb-typescript/base", |     "airbnb-typescript/base", | ||||||
|  |     "plugin:@typescript-eslint/recommended", | ||||||
|     "plugin:wc/recommended", |     "plugin:wc/recommended", | ||||||
|     "plugin:lit/recommended", |     "plugin:lit/recommended", | ||||||
|     "prettier", |     "prettier", | ||||||
| @@ -45,16 +45,16 @@ | |||||||
|     "func-names": 0, |     "func-names": 0, | ||||||
|     "prefer-arrow-callback": 0, |     "prefer-arrow-callback": 0, | ||||||
|     "no-underscore-dangle": 0, |     "no-underscore-dangle": 0, | ||||||
|     "no-var": 0, |  | ||||||
|     "strict": 0, |     "strict": 0, | ||||||
|     "prefer-spread": 0, |     "prefer-spread": 0, | ||||||
|     "no-plusplus": 0, |     "no-plusplus": 0, | ||||||
|     "no-bitwise": 0, |     "no-bitwise": 2, | ||||||
|     "comma-dangle": 0, |     "comma-dangle": 0, | ||||||
|     "vars-on-top": 0, |     "vars-on-top": 0, | ||||||
|     "no-continue": 0, |     "no-continue": 0, | ||||||
|     "no-param-reassign": 0, |     "no-param-reassign": 0, | ||||||
|     "no-multi-assign": 0, |     "no-multi-assign": 0, | ||||||
|  |     "no-console": 2, | ||||||
|     "radix": 0, |     "radix": 0, | ||||||
|     "no-alert": 0, |     "no-alert": 0, | ||||||
|     "no-return-await": 0, |     "no-return-await": 0, | ||||||
| @@ -75,13 +75,16 @@ | |||||||
|     "object-curly-newline": 0, |     "object-curly-newline": 0, | ||||||
|     "default-case": 0, |     "default-case": 0, | ||||||
|     "wc/no-self-class": 0, |     "wc/no-self-class": 0, | ||||||
|  |     "no-shadow": 0, | ||||||
|     "@typescript-eslint/camelcase": 0, |     "@typescript-eslint/camelcase": 0, | ||||||
|     "@typescript-eslint/ban-ts-ignore": 0, |     "@typescript-eslint/ban-ts-comment": 0, | ||||||
|     "@typescript-eslint/no-use-before-define": 0, |     "@typescript-eslint/no-use-before-define": 0, | ||||||
|     "@typescript-eslint/no-non-null-assertion": 0, |     "@typescript-eslint/no-non-null-assertion": 0, | ||||||
|     "@typescript-eslint/no-explicit-any": 0, |     "@typescript-eslint/no-explicit-any": 0, | ||||||
|     "@typescript-eslint/no-unused-vars": 0, |     "@typescript-eslint/no-unused-vars": 0, | ||||||
|     "@typescript-eslint/explicit-function-return-type": 0 |     "@typescript-eslint/explicit-function-return-type": 0, | ||||||
|  |     "@typescript-eslint/explicit-module-boundary-types": 0, | ||||||
|  |     "@typescript-eslint/no-shadow": ["error"] | ||||||
|   }, |   }, | ||||||
|   "plugins": ["disable", "import", "lit", "prettier", "@typescript-eslint"], |   "plugins": ["disable", "import", "lit", "prettier", "@typescript-eslint"], | ||||||
|   "processor": "disable/disable" |   "processor": "disable/disable" | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/BUG_REPORT.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -51,7 +51,7 @@ DO NOT DELETE ANY TEXT from this template! Otherwise, your issue may be closed w | |||||||
| <!-- | <!-- | ||||||
|   Provide details about the versions you are using, which helps us reproducing |   Provide details about the versions you are using, which helps us reproducing | ||||||
|   and finding the issue quicker. Version information is found in the |   and finding the issue quicker. Version information is found in the | ||||||
|   Home Assistant frontend: Developer tools -> Info. |   Home Assistant frontend: Configuration -> Info. | ||||||
|  |  | ||||||
|   Browser version and operating system is important! Please try to replicate |   Browser version and operating system is important! Please try to replicate | ||||||
|   your issue in a different browser and be sure to include your findings. |   your issue in a different browser and be sure to include your findings. | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								.github/ISSUE_TEMPLATE/FEATURE_REQUEST.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,26 +0,0 @@ | |||||||
| --- |  | ||||||
| name: Request a feature for the UI, Frontend or Lovelace |  | ||||||
| about: Request an new feature for the Home Assistant frontend. |  | ||||||
| labels: feature request |  | ||||||
| --- |  | ||||||
|  |  | ||||||
| <!-- |  | ||||||
|   DO NOT DELETE ANY TEXT from this template! |  | ||||||
|   Otherwise, your request may be closed without comment. |  | ||||||
| --> |  | ||||||
|  |  | ||||||
| ## The request |  | ||||||
|  |  | ||||||
| <!-- |  | ||||||
|   Describe to our maintainers, the feature you would like to be added. |  | ||||||
|   Please be clear and concise and, if possible, provide a screenshot or mockup. |  | ||||||
| --> |  | ||||||
|  |  | ||||||
| ## The alternatives |  | ||||||
|  |  | ||||||
| <!-- |  | ||||||
|   Are you currently using, or have you considered alternatives? |  | ||||||
|   If so, could you please describe those? |  | ||||||
| --> |  | ||||||
|  |  | ||||||
| ## Additional information |  | ||||||
							
								
								
									
										3
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,5 +1,8 @@ | |||||||
| blank_issues_enabled: false | blank_issues_enabled: false | ||||||
| contact_links: | contact_links: | ||||||
|  |   - name: Request a feature for the UI, Frontend or Lovelace | ||||||
|  |     url: https://github.com/home-assistant/frontend/discussions/category_choices | ||||||
|  |     about: Request an new feature for the Home Assistant frontend. | ||||||
|   - name: Report a bug that is NOT related to the UI, Frontend or Lovelace |   - name: Report a bug that is NOT related to the UI, Frontend or Lovelace | ||||||
|     url: https://github.com/home-assistant/core/issues |     url: https://github.com/home-assistant/core/issues | ||||||
|     about: This is the issue tracker for our frontend. Please report other issues with the backend repository. |     about: This is the issue tracker for our frontend. Please report other issues with the backend repository. | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -18,8 +18,8 @@ | |||||||
| <!-- | <!-- | ||||||
|   Describe the big picture of your changes here to communicate to the |   Describe the big picture of your changes here to communicate to the | ||||||
|   maintainers why we should accept this pull request. If it fixes a bug |   maintainers why we should accept this pull request. If it fixes a bug | ||||||
|   or resolves a feature request, be sure to link to that issue in the |   or resolves a feature request, be sure to link to that issue or discussion | ||||||
|   additional information section. |   in the additional information section. | ||||||
| --> | --> | ||||||
|  |  | ||||||
| ## Type of change | ## Type of change | ||||||
| @@ -56,7 +56,7 @@ | |||||||
| --> | --> | ||||||
|  |  | ||||||
| - This PR fixes or closes issue: fixes # | - This PR fixes or closes issue: fixes # | ||||||
| - This PR is related to issue: | - This PR is related to issue or discussion: | ||||||
| - Link to documentation pull request: | - Link to documentation pull request: | ||||||
|  |  | ||||||
| ## Checklist | ## Checklist | ||||||
|   | |||||||
							
								
								
									
										27
									
								
								.github/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,27 +0,0 @@ | |||||||
| # Configuration for Lock Threads - https://github.com/dessant/lock-threads |  | ||||||
|  |  | ||||||
| # Number of days of inactivity before a closed issue or pull request is locked |  | ||||||
| daysUntilLock: 1 |  | ||||||
|  |  | ||||||
| # Skip issues and pull requests created before a given timestamp. Timestamp must |  | ||||||
| # follow ISO 8601 (`YYYY-MM-DD`). Set to `false` to disable |  | ||||||
| skipCreatedBefore: 2020-01-01 |  | ||||||
|  |  | ||||||
| # Issues and pull requests with these labels will be ignored. Set to `[]` to disable |  | ||||||
| exemptLabels: [] |  | ||||||
|  |  | ||||||
| # Label to add before locking, such as `outdated`. Set to `false` to disable |  | ||||||
| lockLabel: false |  | ||||||
|  |  | ||||||
| # Comment to post before locking. Set to `false` to disable |  | ||||||
| lockComment: false |  | ||||||
|  |  | ||||||
| # Assign `resolved` as the reason for locking. Set to `false` to disable |  | ||||||
| setLockReason: false |  | ||||||
|  |  | ||||||
| # Limit to only `issues` or `pulls` |  | ||||||
| only: pulls |  | ||||||
|  |  | ||||||
| # Optionally, specify configuration settings just for `issues` or `pulls` |  | ||||||
| issues: |  | ||||||
|   daysUntilLock: 30 |  | ||||||
							
								
								
									
										56
									
								
								.github/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,56 +0,0 @@ | |||||||
| # Configuration for probot-stale - https://github.com/probot/stale |  | ||||||
|  |  | ||||||
| # Number of days of inactivity before an Issue or Pull Request becomes stale |  | ||||||
| daysUntilStale: 90 |  | ||||||
|  |  | ||||||
| # Number of days of inactivity before an Issue or Pull Request with the stale label is closed. |  | ||||||
| # Set to false to disable. If disabled, issues still need to be closed manually, but will remain marked as stale. |  | ||||||
| daysUntilClose: 7 |  | ||||||
|  |  | ||||||
| # Only issues or pull requests with all of these labels are check if stale. Defaults to `[]` (disabled) |  | ||||||
| onlyLabels: [] |  | ||||||
|  |  | ||||||
| # Issues or Pull Requests with these labels will never be considered stale. Set to `[]` to disable |  | ||||||
| exemptLabels: |  | ||||||
|   - feature request |  | ||||||
|   - Help wanted |  | ||||||
|   - to do |  | ||||||
|  |  | ||||||
| # Set to true to ignore issues in a project (defaults to false) |  | ||||||
| exemptProjects: true |  | ||||||
|  |  | ||||||
| # Set to true to ignore issues in a milestone (defaults to false) |  | ||||||
| exemptMilestones: true |  | ||||||
|  |  | ||||||
| # Set to true to ignore issues with an assignee (defaults to false) |  | ||||||
| exemptAssignees: false |  | ||||||
|  |  | ||||||
| # Label to use when marking as stale |  | ||||||
| staleLabel: stale |  | ||||||
|  |  | ||||||
| # Comment to post when marking as stale. Set to `false` to disable |  | ||||||
| markComment: > |  | ||||||
|   There hasn't been any activity on this issue recently. Due to the high number |  | ||||||
|   of incoming GitHub notifications, we have to clean some of the old issues, |  | ||||||
|   as many of them have already been resolved with the latest updates. |  | ||||||
|  |  | ||||||
|   Please make sure to update to the latest Home Assistant version and check |  | ||||||
|   if that solves the issue. Let us know if that works for you by adding a |  | ||||||
|   comment 👍 |  | ||||||
|  |  | ||||||
|   This issue now has been marked as stale and will be closed if no further |  | ||||||
|   activity occurs. Thank you for your contributions. |  | ||||||
|  |  | ||||||
| # Comment to post when removing the stale label. |  | ||||||
| # unmarkComment: > |  | ||||||
| #   Your comment here. |  | ||||||
|  |  | ||||||
| # Comment to post when closing a stale Issue or Pull Request. |  | ||||||
| # closeComment: > |  | ||||||
| #   Your comment here. |  | ||||||
|  |  | ||||||
| # Limit the number of actions per hour, from 1-30. Default is 30 |  | ||||||
| limitPerRun: 30 |  | ||||||
|  |  | ||||||
| # Limit to only `issues` or `pulls` |  | ||||||
| only: issues |  | ||||||
							
								
								
									
										6
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -34,10 +34,8 @@ jobs: | |||||||
|         run: yarn install |         run: yarn install | ||||||
|         env: |         env: | ||||||
|           CI: true |           CI: true | ||||||
|       - name: Build icons |       - name: Build resources | ||||||
|         run: ./node_modules/.bin/gulp gen-icons-json |         run: ./node_modules/.bin/gulp gen-icons-json build-translations gather-gallery-demos | ||||||
|       - name: Build translations |  | ||||||
|         run: ./node_modules/.bin/gulp build-translations |  | ||||||
|       - name: Run eslint |       - name: Run eslint | ||||||
|         run: ./node_modules/.bin/eslint '{**/src,src}/**/*.{js,ts,html}' --ignore-path .gitignore |         run: ./node_modules/.bin/eslint '{**/src,src}/**/*.{js,ts,html}' --ignore-path .gitignore | ||||||
|       - name: Run tsc |       - name: Run tsc | ||||||
|   | |||||||
							
								
								
									
										60
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,60 @@ | |||||||
|  | name: "CodeQL" | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: [dev, master] | ||||||
|  |   pull_request: | ||||||
|  |     # The branches below must be a subset of the branches above | ||||||
|  |     branches: [dev] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   analyze: | ||||||
|  |     name: Analyze | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     strategy: | ||||||
|  |       fail-fast: false | ||||||
|  |       matrix: | ||||||
|  |         # Override automatic language detection by changing the below list | ||||||
|  |         # Supported options are ['csharp', 'cpp', 'go', 'java', 'javascript', 'python'] | ||||||
|  |         language: ['javascript'] | ||||||
|  |         # Learn more... | ||||||
|  |         # https://docs.github.com/en/github/finding-security-vulnerabilities-and-errors-in-your-code/configuring-code-scanning#overriding-automatic-language-detection | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout repository | ||||||
|  |       uses: actions/checkout@v2 | ||||||
|  |       with: | ||||||
|  |         # We must fetch at least the immediate parents so that if this is | ||||||
|  |         # a pull request then we can checkout the head. | ||||||
|  |         fetch-depth: 2 | ||||||
|  |  | ||||||
|  |     # If this run was triggered by a pull request event, then checkout | ||||||
|  |     # the head of the pull request instead of the merge commit. | ||||||
|  |     - run: git checkout HEAD^2 | ||||||
|  |       if: ${{ github.event_name == 'pull_request' }} | ||||||
|  |  | ||||||
|  |     # Initializes the CodeQL tools for scanning. | ||||||
|  |     - name: Initialize CodeQL | ||||||
|  |       uses: github/codeql-action/init@v1 | ||||||
|  |       with: | ||||||
|  |         languages: ${{ matrix.language }} | ||||||
|  |  | ||||||
|  |     # Autobuild attempts to build any compiled languages  (C/C++, C#, or Java). | ||||||
|  |     # If this step fails, then you should remove it and run the build manually (see below) | ||||||
|  |     - name: Autobuild | ||||||
|  |       uses: github/codeql-action/autobuild@v1 | ||||||
|  |  | ||||||
|  |     # ℹ️ Command-line programs to run using the OS shell. | ||||||
|  |     # 📚 https://git.io/JvXDl | ||||||
|  |  | ||||||
|  |     # ✏️ If the Autobuild fails above, remove it and uncomment the following three lines | ||||||
|  |     #    and modify them (or add more) to build your code if your project | ||||||
|  |     #    uses a compiled language | ||||||
|  |  | ||||||
|  |     #- run: | | ||||||
|  |     #   make bootstrap | ||||||
|  |     #   make release | ||||||
|  |  | ||||||
|  |     - name: Perform CodeQL Analysis | ||||||
|  |       uses: github/codeql-action/analyze@v1 | ||||||
							
								
								
									
										20
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | |||||||
|  | name: Lock | ||||||
|  |  | ||||||
|  | # yamllint disable-line rule:truthy | ||||||
|  | on: | ||||||
|  |   schedule: | ||||||
|  |     - cron: "0 * * * *" | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   lock: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: dessant/lock-threads@v2.0.1 | ||||||
|  |         with: | ||||||
|  |           github-token: ${{ github.token }} | ||||||
|  |           issue-lock-inactive-days: "30" | ||||||
|  |           issue-exclude-created-before: "2020-10-01T00:00:00Z" | ||||||
|  |           issue-lock-reason: "" | ||||||
|  |           pr-lock-inactive-days: "1" | ||||||
|  |           pr-exclude-created-before: "2020-11-01T00:00:00Z" | ||||||
|  |           pr-lock-reason: "" | ||||||
							
								
								
									
										14
									
								
								.github/workflows/release-drafter.yaml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | name: Release Drafter | ||||||
|  |  | ||||||
|  | on: | ||||||
|  |   push: | ||||||
|  |     branches: | ||||||
|  |       - dev | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   update_release_draft: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - uses: release-drafter/release-drafter@v5 | ||||||
|  |         env: | ||||||
|  |           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
							
								
								
									
										42
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | |||||||
|  | name: Stale | ||||||
|  |  | ||||||
|  | # yamllint disable-line rule:truthy | ||||||
|  | on: | ||||||
|  |   schedule: | ||||||
|  |     - cron: "0 * * * *" | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   stale: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     steps: | ||||||
|  |       - name: 90 days stale policy | ||||||
|  |         uses: actions/stale@v3.0.13 | ||||||
|  |         with: | ||||||
|  |           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |           days-before-stale: 90 | ||||||
|  |           days-before-close: 7 | ||||||
|  |           operations-per-run: 25 | ||||||
|  |           remove-stale-when-updated: true | ||||||
|  |           stale-issue-label: "stale" | ||||||
|  |           exempt-issue-labels: "no-stale,Help%20wanted,help-wanted,feature-request,feature%20request" | ||||||
|  |           stale-issue-message: > | ||||||
|  |             There hasn't been any activity on this issue recently. Due to the | ||||||
|  |             high number of incoming GitHub notifications, we have to clean some | ||||||
|  |             of the old issues, as many of them have already been resolved with | ||||||
|  |             the latest updates. | ||||||
|  |  | ||||||
|  |             Please make sure to update to the latest Home Assistant version and | ||||||
|  |             check if that solves the issue. Let us know if that works for you by | ||||||
|  |             adding a comment 👍 | ||||||
|  |  | ||||||
|  |             This issue has now been marked as stale and will be closed if no | ||||||
|  |             further activity occurs. Thank you for your contributions. | ||||||
|  |  | ||||||
|  |           stale-pr-label: "stale" | ||||||
|  |           exempt-pr-labels: "no-stale" | ||||||
|  |           stale-pr-message: > | ||||||
|  |             There hasn't been any activity on this pull request recently. This | ||||||
|  |             pull request has been automatically marked as stale because of that | ||||||
|  |             and will be closed if no further activity occurs within 7 days. | ||||||
|  |  | ||||||
|  |             Thank you for your contributions. | ||||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -23,6 +23,8 @@ dist | |||||||
| # vscode | # vscode | ||||||
| .vscode/* | .vscode/* | ||||||
| !.vscode/extensions.json | !.vscode/extensions.json | ||||||
|  | !.vscode/launch.json | ||||||
|  | !.vscode/tasks.json | ||||||
|  |  | ||||||
| # Cast dev settings | # Cast dev settings | ||||||
| src/cast/dev_const.ts | src/cast/dev_const.ts | ||||||
| @@ -33,3 +35,6 @@ yarn-error.log | |||||||
|  |  | ||||||
| #asdf | #asdf | ||||||
| .tool-versions | .tool-versions | ||||||
|  |  | ||||||
|  | # Home Assistant config | ||||||
|  | /config | ||||||
|   | |||||||
							
								
								
									
										44
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,44 @@ | |||||||
|  | { | ||||||
|  |   // https://github.com/microsoft/vscode-js-debug/blob/master/OPTIONS.md | ||||||
|  |   "configurations": [ | ||||||
|  |     { | ||||||
|  |       "name": "Debug Frontend", | ||||||
|  |       "request": "launch", | ||||||
|  |       "type": "pwa-chrome", | ||||||
|  |       "url": "http://localhost:8123/", | ||||||
|  |       "webRoot": "${workspaceFolder}/hass_frontend", | ||||||
|  |       "disableNetworkCache": true, | ||||||
|  |       "preLaunchTask": "Develop Frontend", | ||||||
|  |       "outFiles": [ | ||||||
|  |         "${workspaceFolder}/hass_frontend/frontend_latest/*.js" | ||||||
|  |       ] | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "name": "Debug Gallery", | ||||||
|  |       "request": "launch", | ||||||
|  |       "type": "pwa-chrome", | ||||||
|  |       "url": "http://localhost:8100/", | ||||||
|  |       "webRoot": "${workspaceFolder}/gallery/dist", | ||||||
|  |       "disableNetworkCache": true, | ||||||
|  |       "preLaunchTask": "Develop Gallery" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "name": "Debug Demo", | ||||||
|  |       "request": "launch", | ||||||
|  |       "type": "pwa-chrome", | ||||||
|  |       "url": "http://localhost:8090/", | ||||||
|  |       "webRoot": "${workspaceFolder}/demo/dist", | ||||||
|  |       "disableNetworkCache": true, | ||||||
|  |       "preLaunchTask": "Develop Demo" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "name": "Debug Cast", | ||||||
|  |       "request": "launch", | ||||||
|  |       "type": "pwa-chrome", | ||||||
|  |       "url": "http://localhost:8080/", | ||||||
|  |       "webRoot": "${workspaceFolder}/cast/dist", | ||||||
|  |       "disableNetworkCache": true, | ||||||
|  |       "preLaunchTask": "Develop Cast" | ||||||
|  |     }, | ||||||
|  |   ] | ||||||
|  | } | ||||||
							
								
								
									
										208
									
								
								.vscode/tasks.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,208 @@ | |||||||
|  | { | ||||||
|  |   "version": "2.0.0", | ||||||
|  |   "tasks": [ | ||||||
|  |     { | ||||||
|  |       "label": "Develop Frontend", | ||||||
|  |       "type": "gulp", | ||||||
|  |       "task": "develop-app", | ||||||
|  |       // Sync changes here to other tasks until issue resolved | ||||||
|  |       // https://github.com/Microsoft/vscode/issues/61497 | ||||||
|  |       "problemMatcher": { | ||||||
|  |         "owner": "ha-build", | ||||||
|  |         "source": "ha-build", | ||||||
|  |         "fileLocation": "absolute", | ||||||
|  |         "severity": "error", | ||||||
|  |         "pattern": [ | ||||||
|  |           { | ||||||
|  |             "regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)", | ||||||
|  |             "severity": 1, | ||||||
|  |             "file": 2, | ||||||
|  |             "message": 3, | ||||||
|  |             "line": 4, | ||||||
|  |             "column": 5 | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "background": { | ||||||
|  |           "activeOnStart": true, | ||||||
|  |           "beginsPattern": "Changes detected. Starting compilation", | ||||||
|  |           "endsPattern": "Build done @" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "isBackground": true, | ||||||
|  |       "group": { | ||||||
|  |         "kind": "build", | ||||||
|  |         "isDefault": true | ||||||
|  |       }, | ||||||
|  |       "runOptions": { | ||||||
|  |         "instanceLimit": 1 | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "label": "Develop Supervisor panel", | ||||||
|  |       "type": "gulp", | ||||||
|  |       "task": "develop-hassio", | ||||||
|  |       "problemMatcher": { | ||||||
|  |         "owner": "ha-build", | ||||||
|  |         "source": "ha-build", | ||||||
|  |         "fileLocation": "absolute", | ||||||
|  |         "severity": "error", | ||||||
|  |         "pattern": [ | ||||||
|  |           { | ||||||
|  |             "regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)", | ||||||
|  |             "severity": 1, | ||||||
|  |             "file": 2, | ||||||
|  |             "message": 3, | ||||||
|  |             "line": 4, | ||||||
|  |             "column": 5 | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "background": { | ||||||
|  |           "activeOnStart": true, | ||||||
|  |           "beginsPattern": "Changes detected. Starting compilation", | ||||||
|  |           "endsPattern": "Build done @" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |       "isBackground": true, | ||||||
|  |       "group": "build", | ||||||
|  |       "runOptions": { | ||||||
|  |         "instanceLimit": 1 | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "label": "Develop Gallery", | ||||||
|  |       "type": "gulp", | ||||||
|  |       "task": "develop-gallery", | ||||||
|  |       "problemMatcher": { | ||||||
|  |         "owner": "ha-build", | ||||||
|  |         "source": "ha-build", | ||||||
|  |         "fileLocation": "absolute", | ||||||
|  |         "severity": "error", | ||||||
|  |         "pattern": [ | ||||||
|  |           { | ||||||
|  |             "regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)", | ||||||
|  |             "severity": 1, | ||||||
|  |             "file": 2, | ||||||
|  |             "message": 3, | ||||||
|  |             "line": 4, | ||||||
|  |             "column": 5 | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "background": { | ||||||
|  |           "activeOnStart": true, | ||||||
|  |           "beginsPattern": "Changes detected. Starting compilation", | ||||||
|  |           "endsPattern": "Build done @" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |  | ||||||
|  |       "isBackground": true, | ||||||
|  |       "group": "build", | ||||||
|  |       "runOptions": { | ||||||
|  |         "instanceLimit": 1 | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "label": "Develop Demo", | ||||||
|  |       "type": "gulp", | ||||||
|  |       "task": "develop-demo", | ||||||
|  |       "problemMatcher": { | ||||||
|  |         "owner": "ha-build", | ||||||
|  |         "source": "ha-build", | ||||||
|  |         "fileLocation": "absolute", | ||||||
|  |         "severity": "error", | ||||||
|  |         "pattern": [ | ||||||
|  |           { | ||||||
|  |             "regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)", | ||||||
|  |             "severity": 1, | ||||||
|  |             "file": 2, | ||||||
|  |             "message": 3, | ||||||
|  |             "line": 4, | ||||||
|  |             "column": 5 | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "background": { | ||||||
|  |           "activeOnStart": true, | ||||||
|  |           "beginsPattern": "Changes detected. Starting compilation", | ||||||
|  |           "endsPattern": "Build done @" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |  | ||||||
|  |       "isBackground": true, | ||||||
|  |       "group": "build", | ||||||
|  |       "runOptions": { | ||||||
|  |         "instanceLimit": 1 | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "label": "Develop Cast", | ||||||
|  |       "type": "gulp", | ||||||
|  |       "task": "develop-cast", | ||||||
|  |       "problemMatcher": { | ||||||
|  |         "owner": "ha-build", | ||||||
|  |         "source": "ha-build", | ||||||
|  |         "fileLocation": "absolute", | ||||||
|  |         "severity": "error", | ||||||
|  |         "pattern": [ | ||||||
|  |           { | ||||||
|  |             "regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)", | ||||||
|  |             "severity": 1, | ||||||
|  |             "file": 2, | ||||||
|  |             "message": 3, | ||||||
|  |             "line": 4, | ||||||
|  |             "column": 5 | ||||||
|  |           } | ||||||
|  |         ], | ||||||
|  |         "background": { | ||||||
|  |           "activeOnStart": true, | ||||||
|  |           "beginsPattern": "Changes detected. Starting compilation", | ||||||
|  |           "endsPattern": "Build done @" | ||||||
|  |         } | ||||||
|  |       }, | ||||||
|  |  | ||||||
|  |       "isBackground": true, | ||||||
|  |       "group": "build", | ||||||
|  |       "runOptions": { | ||||||
|  |         "instanceLimit": 1 | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "label": "Run HA Core in devcontainer", | ||||||
|  |       "type": "shell", | ||||||
|  |       "command": "script/core", | ||||||
|  |       "isBackground": true, | ||||||
|  |       "group": { | ||||||
|  |         "kind": "build", | ||||||
|  |         "isDefault": true | ||||||
|  |       }, | ||||||
|  |       "problemMatcher": [], | ||||||
|  |       "runOptions": { | ||||||
|  |         "instanceLimit": 1 | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "label": "Run HA Core for Supervisor in devcontainer", | ||||||
|  |       "type": "shell", | ||||||
|  |       "command": "HASSIO=${input:supervisorHost} HASSIO_TOKEN=${input:supervisorToken} script/core", | ||||||
|  |       "isBackground": true, | ||||||
|  |       "group": { | ||||||
|  |         "kind": "build", | ||||||
|  |         "isDefault": true | ||||||
|  |       }, | ||||||
|  |       "problemMatcher": [], | ||||||
|  |       "runOptions": { | ||||||
|  |         "instanceLimit": 1 | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   "inputs": [ | ||||||
|  |     { | ||||||
|  |       "id": "supervisorHost", | ||||||
|  |       "type": "promptString", | ||||||
|  |       "description": "The IP of the Supervisor host running the Remote API proxy add-on" | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       "id": "supervisorToken", | ||||||
|  |       "type": "promptString", | ||||||
|  |       "description": "The token for the Remote API proxy add-on" | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
| @@ -2,79 +2,139 @@ | |||||||
|  |  | ||||||
| ## Our Pledge | ## Our Pledge | ||||||
|  |  | ||||||
| In the interest of fostering an open and welcoming environment, we as | We as members, contributors, and leaders pledge to make participation in our | ||||||
| contributors and maintainers pledge to making participation in our project and | community a harassment-free experience for everyone, regardless of age, body | ||||||
| our community a harassment-free experience for everyone, regardless of age, body | size, visible or invisible disability, ethnicity, sex characteristics, gender | ||||||
| size, disability, ethnicity, gender identity and expression, level of experience, | identity and expression, level of experience, education, socio-economic status, | ||||||
| nationality, personal appearance, race, religion, or sexual identity and | nationality, personal appearance, race, religion, or sexual identity | ||||||
| orientation. | and orientation. | ||||||
|  |  | ||||||
|  | We pledge to act and interact in ways that contribute to an open, welcoming, | ||||||
|  | diverse, inclusive, and healthy community. | ||||||
|  |  | ||||||
| ## Our Standards | ## Our Standards | ||||||
|  |  | ||||||
| Examples of behavior that contributes to creating a positive environment | Examples of behavior that contributes to a positive environment for our | ||||||
| include: | community include: | ||||||
|  |  | ||||||
| * Using welcoming and inclusive language | * Demonstrating empathy and kindness toward other people | ||||||
| * Being respectful of differing viewpoints and experiences | * Being respectful of differing opinions, viewpoints, and experiences | ||||||
| * Gracefully accepting constructive criticism | * Giving and gracefully accepting constructive feedback | ||||||
| * Focusing on what is best for the community | * Accepting responsibility and apologizing to those affected by our mistakes, | ||||||
| * Showing empathy towards other community members |   and learning from the experience | ||||||
|  | * Focusing on what is best not just for us as individuals, but for the | ||||||
|  |   overall community | ||||||
|  |  | ||||||
| Examples of unacceptable behavior by participants include: | Examples of unacceptable behavior include: | ||||||
|  |  | ||||||
| * The use of sexualized language or imagery and unwelcome sexual attention or | * The use of sexualized language or imagery, and sexual attention or | ||||||
| advances |   advances of any kind | ||||||
| * Trolling, insulting/derogatory comments, and personal or political attacks | * Trolling, insulting or derogatory comments, and personal or political attacks | ||||||
| * Public or private harassment | * Public or private harassment | ||||||
| * Publishing others' private information, such as a physical or electronic | * Publishing others' private information, such as a physical or email | ||||||
|   address, without explicit permission |   address, without their explicit permission | ||||||
| * Other conduct which could reasonably be considered inappropriate in a | * Other conduct which could reasonably be considered inappropriate in a | ||||||
|   professional setting |   professional setting | ||||||
|  |  | ||||||
| ## Our Responsibilities | ## Enforcement Responsibilities | ||||||
|  |  | ||||||
| Project maintainers are responsible for clarifying the standards of acceptable | Community leaders are responsible for clarifying and enforcing our standards of | ||||||
| behavior and are expected to take appropriate and fair corrective action in | acceptable behavior and will take appropriate and fair corrective action in | ||||||
| response to any instances of unacceptable behavior. | response to any behavior that they deem inappropriate, threatening, offensive, | ||||||
|  | or harmful. | ||||||
|  |  | ||||||
| Project maintainers have the right and responsibility to remove, edit, or | Community leaders have the right and responsibility to remove, edit, or reject | ||||||
| reject comments, commits, code, wiki edits, issues, and other contributions | comments, commits, code, wiki edits, issues, and other contributions that are | ||||||
| that are not aligned to this Code of Conduct, or to ban temporarily or | not aligned to this Code of Conduct, and will communicate reasons for moderation | ||||||
| permanently any contributor for other behaviors that they deem inappropriate, | decisions when appropriate. | ||||||
| threatening, offensive, or harmful. |  | ||||||
|  |  | ||||||
| ## Scope | ## Scope | ||||||
|  |  | ||||||
| This Code of Conduct applies both within project spaces and in public spaces | This Code of Conduct applies within all community spaces, and also applies when | ||||||
| when an individual is representing the project or its community. Examples of | an individual is officially representing the community in public spaces. | ||||||
| representing a project or community include using an official project e-mail | Examples of representing our community include using an official e-mail address, | ||||||
| address, posting via an official social media account, or acting as an appointed | posting via an official social media account, or acting as an appointed | ||||||
| representative at an online or offline event. Representation of a project may be | representative at an online or offline event. | ||||||
| further defined and clarified by project maintainers. |  | ||||||
|  |  | ||||||
| ## Enforcement | ## Enforcement | ||||||
|  |  | ||||||
| Instances of abusive, harassing, or otherwise unacceptable behavior may be | Instances of abusive, harassing, or otherwise unacceptable behavior may be | ||||||
| reported by contacting the project team at [safety@home-assistant.io][email]. All | reported to the community leaders responsible for enforcement at | ||||||
| complaints will be reviewed and investigated and will result in a response that | [safety@home-assistant.io][email] or by using the report/flag feature of | ||||||
| is deemed necessary and appropriate to the circumstances. The project team is | the medium used. All complaints will be reviewed and investigated promptly and | ||||||
| obligated to maintain confidentiality with regard to the reporter of an incident. | fairly. | ||||||
| Further details of specific enforcement policies may be posted separately. |  | ||||||
|  |  | ||||||
| Project maintainers who do not follow or enforce the Code of Conduct in good | All community leaders are obligated to respect the privacy and security of the | ||||||
| faith may face temporary or permanent repercussions as determined by other | reporter of any incident. | ||||||
| members of the project's leadership. |  | ||||||
|  | ## Enforcement Guidelines | ||||||
|  |  | ||||||
|  | Community leaders will follow these Community Impact Guidelines in determining | ||||||
|  | the consequences for any action they deem in violation of this Code of Conduct: | ||||||
|  |  | ||||||
|  | ### 1. Correction | ||||||
|  |  | ||||||
|  | **Community Impact**: Use of inappropriate language or other behavior deemed | ||||||
|  | unprofessional or unwelcome in the community. | ||||||
|  |  | ||||||
|  | **Consequence**: A private, written warning from community leaders, providing | ||||||
|  | clarity around the nature of the violation and an explanation of why the | ||||||
|  | behavior was inappropriate. A public apology may be requested. | ||||||
|  |  | ||||||
|  | ### 2. Warning | ||||||
|  |  | ||||||
|  | **Community Impact**: A violation through a single incident or series | ||||||
|  | of actions. | ||||||
|  |  | ||||||
|  | **Consequence**: A warning with consequences for continued behavior. No | ||||||
|  | interaction with the people involved, including unsolicited interaction with | ||||||
|  | those enforcing the Code of Conduct, for a specified period of time. This | ||||||
|  | includes avoiding interactions in community spaces as well as external channels | ||||||
|  | like social media. Violating these terms may lead to a temporary or | ||||||
|  | permanent ban. | ||||||
|  |  | ||||||
|  | ### 3. Temporary Ban | ||||||
|  |  | ||||||
|  | **Community Impact**: A serious violation of community standards, including | ||||||
|  | sustained inappropriate behavior. | ||||||
|  |  | ||||||
|  | **Consequence**: A temporary ban from any sort of interaction or public | ||||||
|  | communication with the community for a specified period of time. No public or | ||||||
|  | private interaction with the people involved, including unsolicited interaction | ||||||
|  | with those enforcing the Code of Conduct, is allowed during this period. | ||||||
|  | Violating these terms may lead to a permanent ban. | ||||||
|  |  | ||||||
|  | ### 4. Permanent Ban | ||||||
|  |  | ||||||
|  | **Community Impact**: Demonstrating a pattern of violation of community | ||||||
|  | standards, including sustained inappropriate behavior,  harassment of an | ||||||
|  | individual, or aggression toward or disparagement of classes of individuals. | ||||||
|  |  | ||||||
|  | **Consequence**: A permanent ban from any sort of public interaction within | ||||||
|  | the community. | ||||||
|  |  | ||||||
| ## Attribution | ## Attribution | ||||||
|  |  | ||||||
| This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4, | This Code of Conduct is adapted from the [Contributor Covenant][homepage], | ||||||
| available [here][version]. | version 2.0, available [here][version]. | ||||||
|  |  | ||||||
|  | Community Impact Guidelines were inspired by [Mozilla's code of conduct | ||||||
|  | enforcement ladder][mozilla]. | ||||||
|  |  | ||||||
| ## Adoption | ## Adoption | ||||||
|  |  | ||||||
| This Code of Conduct was first adopted January 21st, 2017 and announced in [this][coc-blog] blog post. | This Code of Conduct was first adopted January 21st, 2017 and announced in | ||||||
|  | [this][coc-blog] blog post and has been updated on May 25th, 2020 to version | ||||||
|  | 2.0 of the [Contributor Covenant][homepage] as announced in [this][coc2-blog] | ||||||
|  | blog post. | ||||||
|  |  | ||||||
| [homepage]: http://contributor-covenant.org | For answers to common questions about this code of conduct, see the FAQ at | ||||||
| [version]: http://contributor-covenant.org/version/1/4/ | <https://www.contributor-covenant.org/faq>. Translations are available at | ||||||
|  | <https://www.contributor-covenant.org/translations>. | ||||||
|  |  | ||||||
|  | [coc-blog]: /blog/2017/01/21/home-assistant-governance/ | ||||||
|  | [coc2-blog]: /blog/2020/05/25/code-of-conduct-updated/ | ||||||
| [email]: mailto:safety@home-assistant.io | [email]: mailto:safety@home-assistant.io | ||||||
| [coc-blog]: https://home-assistant.io/blog/2017/01/21/home-assistant-governance/ | [homepage]: http://contributor-covenant.org | ||||||
|  | [mozilla]: https://github.com/mozilla/diversity | ||||||
|  | [version]: https://www.contributor-covenant.org/version/2/0/code_of_conduct.html | ||||||
|   | |||||||
							
								
								
									
										31
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						| @@ -1,31 +0,0 @@ | |||||||
| FROM node:8.11.1-alpine |  | ||||||
|  |  | ||||||
| # install yarn |  | ||||||
| ENV PATH /root/.yarn/bin:$PATH |  | ||||||
|  |  | ||||||
| ## Install/force base tools |  | ||||||
| RUN apk update \ |  | ||||||
|   && apk add make g++ curl bash binutils tar git python2 python3 \ |  | ||||||
|   && rm -rf /var/cache/apk/* \ |  | ||||||
|   && /bin/bash \ |  | ||||||
|   && touch ~/.bashrc |  | ||||||
|  |  | ||||||
| ## Install yarn |  | ||||||
| RUN curl -o- -L https://yarnpkg.com/install.sh | bash |  | ||||||
|  |  | ||||||
| ## Setup the project |  | ||||||
| RUN mkdir -p /frontend |  | ||||||
|  |  | ||||||
| WORKDIR /frontend |  | ||||||
|  |  | ||||||
| COPY package.json yarn.lock ./ |  | ||||||
|  |  | ||||||
| RUN yarn install --frozen-lockfile |  | ||||||
|  |  | ||||||
| COPY . . |  | ||||||
|  |  | ||||||
| COPY script/docker_entrypoint.sh /usr/bin/docker_entrypoint.sh |  | ||||||
|  |  | ||||||
| RUN chmod +x /usr/bin/docker_entrypoint.sh |  | ||||||
|  |  | ||||||
| CMD [ "docker_entrypoint.sh" ] |  | ||||||
							
								
								
									
										11
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -22,17 +22,8 @@ This is the repository for the official [Home Assistant](https://home-assistant. | |||||||
|  |  | ||||||
| A complete guide can be found at the following [link](https://www.home-assistant.io/developers/frontend/). It describes a short guide for the build of project. | A complete guide can be found at the following [link](https://www.home-assistant.io/developers/frontend/). It describes a short guide for the build of project. | ||||||
|  |  | ||||||
| ### Docker environment |  | ||||||
|  |  | ||||||
| It is possible to compile the project and/or run commands in the development environment having only the [Docker](https://www.docker.com) pre-installed in the system. On the root of project you can do: |  | ||||||
|  |  | ||||||
| - `sh ./script/docker_run.sh build` Build all the project with one command |  | ||||||
| - `sh ./script/docker_run.sh bash` Open an interactive shell (the same environment generated by the _classic environment_) where you can run commands. This bash work on your project directory and any change on your file is automatically present within your build bash. |  | ||||||
|  |  | ||||||
| **Note**: if you have installed `npm` in addition to the `docker`, you can use the commands `npm run docker_build` and `npm run bash` to get a full build or bash as explained above |  | ||||||
|  |  | ||||||
| ## License | ## License | ||||||
|  |  | ||||||
| Home Assistant is open-source and Apache 2 licensed. Feel free to browse the repository, learn and reuse parts in your own projects. | Home Assistant is open-source and Apache 2 licensed. Feel free to browse the repository, learn and reuse parts in your own projects. | ||||||
|  |  | ||||||
| We use [BrowserStack](https://www.browserstack.com) to test Home Assistant on a large variation of devices. | We use [BrowserStack](https://www.browserstack.com) to test Home Assistant on a large variety of devices. | ||||||
|   | |||||||
| @@ -1,39 +0,0 @@ | |||||||
| const options = ({ latestBuild }) => ({ |  | ||||||
|   presets: [ |  | ||||||
|     !latestBuild && [require("@babel/preset-env").default, { modules: false }], |  | ||||||
|     require("@babel/preset-typescript").default, |  | ||||||
|   ].filter(Boolean), |  | ||||||
|   plugins: [ |  | ||||||
|     // Part of ES2018. Converts {...a, b: 2} to Object.assign({}, a, {b: 2}) |  | ||||||
|     [ |  | ||||||
|       "@babel/plugin-proposal-object-rest-spread", |  | ||||||
|       { loose: true, useBuiltIns: true }, |  | ||||||
|     ], |  | ||||||
|     // Only support the syntax, Webpack will handle it. |  | ||||||
|     "@babel/syntax-dynamic-import", |  | ||||||
|     "@babel/plugin-proposal-optional-chaining", |  | ||||||
|     "@babel/plugin-proposal-nullish-coalescing-operator", |  | ||||||
|     [ |  | ||||||
|       require("@babel/plugin-proposal-decorators").default, |  | ||||||
|       { decoratorsBeforeExport: true }, |  | ||||||
|     ], |  | ||||||
|     [ |  | ||||||
|       require("@babel/plugin-proposal-class-properties").default, |  | ||||||
|       { loose: true }, |  | ||||||
|     ], |  | ||||||
|   ], |  | ||||||
| }); |  | ||||||
|  |  | ||||||
| module.exports.babelLoaderConfig = ({ latestBuild }) => { |  | ||||||
|   if (latestBuild === undefined) { |  | ||||||
|     throw Error("latestBuild not defined for babel loader config"); |  | ||||||
|   } |  | ||||||
|   return { |  | ||||||
|     test: /\.m?js$|\.tsx?$/, |  | ||||||
|     exclude: [require.resolve("@mdi/js/mdi.js"), require.resolve("hls.js")], |  | ||||||
|     use: { |  | ||||||
|       loader: "babel-loader", |  | ||||||
|       options: options({ latestBuild }), |  | ||||||
|     }, |  | ||||||
|   }; |  | ||||||
| }; |  | ||||||
							
								
								
									
										202
									
								
								build-scripts/bundle.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,202 @@ | |||||||
|  | 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"), | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | // Files from NPM packages that we should replace with empty file | ||||||
|  | module.exports.emptyPackages = ({ latestBuild }) => | ||||||
|  |   [ | ||||||
|  |     // Contains all color definitions for all material color sets. | ||||||
|  |     // We don't use it | ||||||
|  |     require.resolve("@polymer/paper-styles/color.js"), | ||||||
|  |     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"), | ||||||
|  |     // Compatibility not needed for latest builds | ||||||
|  |     latestBuild && | ||||||
|  |       // wrapped in require.resolve so it blows up if file no longer exists | ||||||
|  |       require.resolve( | ||||||
|  |         path.resolve(paths.polymer_dir, "src/resources/compatibility.ts") | ||||||
|  |       ), | ||||||
|  |     // This polyfill is loaded in workers to support ES5, filter it out. | ||||||
|  |     latestBuild && require.resolve("proxy-polyfill/src/index.js"), | ||||||
|  |   ].filter(Boolean); | ||||||
|  |  | ||||||
|  | module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({ | ||||||
|  |   __DEV__: !isProdBuild, | ||||||
|  |   __BUILD__: JSON.stringify(latestBuild ? "latest" : "es5"), | ||||||
|  |   __VERSION__: JSON.stringify(env.version()), | ||||||
|  |   __DEMO__: false, | ||||||
|  |   __BACKWARDS_COMPAT__: false, | ||||||
|  |   __STATIC_PATH__: "/static/", | ||||||
|  |   "process.env.NODE_ENV": JSON.stringify( | ||||||
|  |     isProdBuild ? "production" : "development" | ||||||
|  |   ), | ||||||
|  |   ...defineOverlay, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports.terserOptions = (latestBuild) => ({ | ||||||
|  |   safari10: true, | ||||||
|  |   ecma: latestBuild ? undefined : 5, | ||||||
|  |   output: { comments: false }, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports.babelOptions = ({ latestBuild }) => ({ | ||||||
|  |   babelrc: false, | ||||||
|  |   presets: [ | ||||||
|  |     !latestBuild && [ | ||||||
|  |       require("@babel/preset-env").default, | ||||||
|  |       { | ||||||
|  |         useBuiltIns: "entry", | ||||||
|  |         corejs: "3.6", | ||||||
|  |       }, | ||||||
|  |     ], | ||||||
|  |     require("@babel/preset-typescript").default, | ||||||
|  |   ].filter(Boolean), | ||||||
|  |   plugins: [ | ||||||
|  |     // Part of ES2018. Converts {...a, b: 2} to Object.assign({}, a, {b: 2}) | ||||||
|  |     !latestBuild && [ | ||||||
|  |       "@babel/plugin-proposal-object-rest-spread", | ||||||
|  |       { loose: true, useBuiltIns: true }, | ||||||
|  |     ], | ||||||
|  |     // Only support the syntax, Webpack will handle it. | ||||||
|  |     "@babel/plugin-syntax-import-meta", | ||||||
|  |     "@babel/plugin-syntax-dynamic-import", | ||||||
|  |     "@babel/plugin-proposal-optional-chaining", | ||||||
|  |     "@babel/plugin-proposal-nullish-coalescing-operator", | ||||||
|  |     [ | ||||||
|  |       require("@babel/plugin-proposal-decorators").default, | ||||||
|  |       { decoratorsBeforeExport: true }, | ||||||
|  |     ], | ||||||
|  |     [ | ||||||
|  |       require("@babel/plugin-proposal-class-properties").default, | ||||||
|  |       { loose: true }, | ||||||
|  |     ], | ||||||
|  |   ].filter(Boolean), | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Are already ES5, cause warnings when babelified. | ||||||
|  | module.exports.babelExclude = () => [ | ||||||
|  |   require.resolve("@mdi/js/mdi.js"), | ||||||
|  |   require.resolve("hls.js"), | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | const outputPath = (outputRoot, latestBuild) => | ||||||
|  |   path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5"); | ||||||
|  |  | ||||||
|  | const publicPath = (latestBuild, root = "") => | ||||||
|  |   latestBuild ? `${root}/frontend_latest/` : `${root}/frontend_es5/`; | ||||||
|  |  | ||||||
|  | /* | ||||||
|  | BundleConfig { | ||||||
|  |   // Object with entrypoints that need to be bundled | ||||||
|  |   entry: { [name: string]: pathToFile }, | ||||||
|  |   // Folder where bundled files need to be written | ||||||
|  |   outputPath: string, | ||||||
|  |   // absolute url-path where bundled files can be found | ||||||
|  |   publicPath: string, | ||||||
|  |   // extra definitions that we need to replace in source | ||||||
|  |   defineOverlay: {[name: string]: value }, | ||||||
|  |   // if this is a production build | ||||||
|  |   isProdBuild: boolean, | ||||||
|  |   // If we're targeting latest browsers | ||||||
|  |   latestBuild: boolean, | ||||||
|  |   // If we're doing a stats build (create nice chunk names) | ||||||
|  |   isStatsBuild: boolean, | ||||||
|  |   // Names of entrypoints that should not be hashed | ||||||
|  |   dontHash: Set<string> | ||||||
|  | } | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | module.exports.config = { | ||||||
|  |   app({ isProdBuild, latestBuild, isStatsBuild }) { | ||||||
|  |     return { | ||||||
|  |       entry: { | ||||||
|  |         service_worker: "./src/entrypoints/service_worker.ts", | ||||||
|  |         app: "./src/entrypoints/app.ts", | ||||||
|  |         authorize: "./src/entrypoints/authorize.ts", | ||||||
|  |         onboarding: "./src/entrypoints/onboarding.ts", | ||||||
|  |         core: "./src/entrypoints/core.ts", | ||||||
|  |         "custom-panel": "./src/entrypoints/custom-panel.ts", | ||||||
|  |       }, | ||||||
|  |       outputPath: outputPath(paths.app_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild), | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       isStatsBuild, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   demo({ isProdBuild, latestBuild, isStatsBuild }) { | ||||||
|  |     return { | ||||||
|  |       entry: { | ||||||
|  |         main: path.resolve(paths.demo_dir, "src/entrypoint.ts"), | ||||||
|  |       }, | ||||||
|  |       outputPath: outputPath(paths.demo_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild), | ||||||
|  |       defineOverlay: { | ||||||
|  |         __VERSION__: JSON.stringify(`DEMO-${env.version()}`), | ||||||
|  |         __DEMO__: true, | ||||||
|  |       }, | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       isStatsBuild, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   cast({ isProdBuild, latestBuild }) { | ||||||
|  |     const entry = { | ||||||
|  |       launcher: path.resolve(paths.cast_dir, "src/launcher/entrypoint.ts"), | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     if (latestBuild) { | ||||||
|  |       entry.receiver = path.resolve( | ||||||
|  |         paths.cast_dir, | ||||||
|  |         "src/receiver/entrypoint.ts" | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return { | ||||||
|  |       entry, | ||||||
|  |       outputPath: outputPath(paths.cast_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild), | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       defineOverlay: { | ||||||
|  |         __BACKWARDS_COMPAT__: true, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   hassio({ isProdBuild, latestBuild }) { | ||||||
|  |     return { | ||||||
|  |       entry: { | ||||||
|  |         entrypoint: path.resolve(paths.hassio_dir, "src/entrypoint.ts"), | ||||||
|  |       }, | ||||||
|  |       outputPath: outputPath(paths.hassio_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild, paths.hassio_publicPath), | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  |  | ||||||
|  |   gallery({ isProdBuild, latestBuild }) { | ||||||
|  |     return { | ||||||
|  |       entry: { | ||||||
|  |         entrypoint: path.resolve(paths.gallery_dir, "src/entrypoint.js"), | ||||||
|  |       }, | ||||||
|  |       outputPath: outputPath(paths.gallery_output_root, latestBuild), | ||||||
|  |       publicPath: publicPath(latestBuild), | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  | }; | ||||||
| @@ -3,8 +3,13 @@ const path = require("path"); | |||||||
| const paths = require("./paths.js"); | const paths = require("./paths.js"); | ||||||
|  |  | ||||||
| module.exports = { | module.exports = { | ||||||
|  |   useRollup() { | ||||||
|  |     return process.env.ROLLUP === "1"; | ||||||
|  |   }, | ||||||
|   isProdBuild() { |   isProdBuild() { | ||||||
|     return process.env.NODE_ENV === "production"; |     return ( | ||||||
|  |       process.env.NODE_ENV === "production" || module.exports.isStatsBuild() | ||||||
|  |     ); | ||||||
|   }, |   }, | ||||||
|   isStatsBuild() { |   isStatsBuild() { | ||||||
|     return process.env.STATS === "1"; |     return process.env.STATS === "1"; | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| // Run HA develop mode | // Run HA develop mode | ||||||
| const gulp = require("gulp"); | const gulp = require("gulp"); | ||||||
|  |  | ||||||
| const envVars = require("../env"); | const env = require("../env"); | ||||||
|  |  | ||||||
| require("./clean.js"); | require("./clean.js"); | ||||||
| require("./translations.js"); | require("./translations.js"); | ||||||
| @@ -11,6 +11,7 @@ require("./compress.js"); | |||||||
| require("./webpack.js"); | require("./webpack.js"); | ||||||
| require("./service-worker.js"); | require("./service-worker.js"); | ||||||
| require("./entry-html.js"); | require("./entry-html.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-app", |   "develop-app", | ||||||
| @@ -26,8 +27,8 @@ gulp.task( | |||||||
|       "gen-index-app-dev", |       "gen-index-app-dev", | ||||||
|       "build-translations" |       "build-translations" | ||||||
|     ), |     ), | ||||||
|     "copy-static", |     "copy-static-app", | ||||||
|     "webpack-watch-app" |     env.useRollup() ? "rollup-watch-app" : "webpack-watch-app" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -39,10 +40,10 @@ gulp.task( | |||||||
|     }, |     }, | ||||||
|     "clean", |     "clean", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations"), |     gulp.parallel("gen-icons-json", "build-translations"), | ||||||
|     "copy-static", |     "copy-static-app", | ||||||
|     "webpack-prod-app", |     env.useRollup() ? "rollup-prod-app" : "webpack-prod-app", | ||||||
|     ...// Don't compress running tests |     ...// Don't compress running tests | ||||||
|     (envVars.isTest() ? [] : ["compress-app"]), |     (env.isTest() ? [] : ["compress-app"]), | ||||||
|     gulp.parallel( |     gulp.parallel( | ||||||
|       "gen-pages-prod", |       "gen-pages-prod", | ||||||
|       "gen-index-app-prod", |       "gen-index-app-prod", | ||||||
|   | |||||||
| @@ -1,11 +1,14 @@ | |||||||
| const gulp = require("gulp"); | const gulp = require("gulp"); | ||||||
|  |  | ||||||
|  | const env = require("../env"); | ||||||
|  |  | ||||||
| require("./clean.js"); | require("./clean.js"); | ||||||
| require("./translations.js"); | require("./translations.js"); | ||||||
| require("./gather-static.js"); | require("./gather-static.js"); | ||||||
| require("./webpack.js"); | require("./webpack.js"); | ||||||
| require("./service-worker.js"); | require("./service-worker.js"); | ||||||
| require("./entry-html.js"); | require("./entry-html.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-cast", |   "develop-cast", | ||||||
| @@ -17,7 +20,8 @@ gulp.task( | |||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations"), |     gulp.parallel("gen-icons-json", "build-translations"), | ||||||
|     "copy-static-cast", |     "copy-static-cast", | ||||||
|     "webpack-dev-server-cast" |     "gen-index-cast-dev", | ||||||
|  |     env.useRollup() ? "rollup-dev-server-cast" : "webpack-dev-server-cast" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -31,7 +35,7 @@ gulp.task( | |||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations"), |     gulp.parallel("gen-icons-json", "build-translations"), | ||||||
|     "copy-static-cast", |     "copy-static-cast", | ||||||
|     "webpack-prod-cast", |     env.useRollup() ? "rollup-prod-cast" : "webpack-prod-cast", | ||||||
|     "gen-index-cast-prod" |     "gen-index-cast-prod" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -1,39 +1,36 @@ | |||||||
| const del = require("del"); | const del = require("del"); | ||||||
| const gulp = require("gulp"); | const gulp = require("gulp"); | ||||||
| const config = require("../paths"); | const paths = require("../paths"); | ||||||
| require("./translations"); | require("./translations"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "clean", |   "clean", | ||||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { |   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||||
|     return del([config.root, config.build_dir]); |     return del([paths.app_output_root, paths.build_dir]); | ||||||
|   }) |   }) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "clean-demo", |   "clean-demo", | ||||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { |   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||||
|     return del([config.demo_root, config.build_dir]); |     return del([paths.demo_output_root, paths.build_dir]); | ||||||
|   }) |   }) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "clean-cast", |   "clean-cast", | ||||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { |   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||||
|     return del([config.cast_root, config.build_dir]); |     return del([paths.cast_output_root, paths.build_dir]); | ||||||
|   }) |   }) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task("clean-hassio", function cleanOutputAndBuildDir() { | ||||||
|   "clean-hassio", |   return del([paths.hassio_output_root, paths.build_dir]); | ||||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | }); | ||||||
|     return del([config.hassio_root, config.build_dir]); |  | ||||||
|   }) |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "clean-gallery", |   "clean-gallery", | ||||||
|   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { |   gulp.parallel("clean-translations", function cleanOutputAndBuildDir() { | ||||||
|     return del([config.gallery_root, config.build_dir]); |     return del([paths.gallery_output_root, paths.build_dir]); | ||||||
|   }) |   }) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -6,38 +6,40 @@ const merge = require("merge-stream"); | |||||||
| const path = require("path"); | const path = require("path"); | ||||||
| const paths = require("../paths"); | const paths = require("../paths"); | ||||||
|  |  | ||||||
|  | const zopfliOptions = { threshold: 150 }; | ||||||
|  |  | ||||||
| gulp.task("compress-app", function compressApp() { | gulp.task("compress-app", function compressApp() { | ||||||
|   const jsLatest = gulp |   const jsLatest = gulp | ||||||
|     .src(path.resolve(paths.output, "**/*.js")) |     .src(path.resolve(paths.app_output_latest, "**/*.js")) | ||||||
|     .pipe(zopfli({ threshold: 150 })) |     .pipe(zopfli(zopfliOptions)) | ||||||
|     .pipe(gulp.dest(paths.output)); |     .pipe(gulp.dest(paths.app_output_latest)); | ||||||
|  |  | ||||||
|   const jsEs5 = gulp |   const jsEs5 = gulp | ||||||
|     .src(path.resolve(paths.output_es5, "**/*.js")) |     .src(path.resolve(paths.app_output_es5, "**/*.js")) | ||||||
|     .pipe(zopfli({ threshold: 150 })) |     .pipe(zopfli(zopfliOptions)) | ||||||
|     .pipe(gulp.dest(paths.output_es5)); |     .pipe(gulp.dest(paths.app_output_es5)); | ||||||
|  |  | ||||||
|   const polyfills = gulp |   const polyfills = gulp | ||||||
|     .src(path.resolve(paths.static, "polyfills/*.js")) |     .src(path.resolve(paths.app_output_static, "polyfills/*.js")) | ||||||
|     .pipe(zopfli({ threshold: 150 })) |     .pipe(zopfli(zopfliOptions)) | ||||||
|     .pipe(gulp.dest(path.resolve(paths.static, "polyfills"))); |     .pipe(gulp.dest(path.resolve(paths.app_output_static, "polyfills"))); | ||||||
|  |  | ||||||
|   const translations = gulp |   const translations = gulp | ||||||
|     .src(path.resolve(paths.static, "translations/**/*.json")) |     .src(path.resolve(paths.app_output_static, "translations/**/*.json")) | ||||||
|     .pipe(zopfli({ threshold: 150 })) |     .pipe(zopfli(zopfliOptions)) | ||||||
|     .pipe(gulp.dest(path.resolve(paths.static, "translations"))); |     .pipe(gulp.dest(path.resolve(paths.app_output_static, "translations"))); | ||||||
|  |  | ||||||
|   const icons = gulp |   const icons = gulp | ||||||
|     .src(path.resolve(paths.static, "mdi/*.json")) |     .src(path.resolve(paths.app_output_static, "mdi/*.json")) | ||||||
|     .pipe(zopfli({ threshold: 150 })) |     .pipe(zopfli(zopfliOptions)) | ||||||
|     .pipe(gulp.dest(path.resolve(paths.static, "mdi"))); |     .pipe(gulp.dest(path.resolve(paths.app_output_static, "mdi"))); | ||||||
|  |  | ||||||
|   return merge(jsLatest, jsEs5, polyfills, translations, icons); |   return merge(jsLatest, jsEs5, polyfills, translations, icons); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("compress-hassio", function compressApp() { | gulp.task("compress-hassio", function compressApp() { | ||||||
|   return gulp |   return gulp | ||||||
|     .src(path.resolve(paths.hassio_root, "**/*.js")) |     .src(path.resolve(paths.hassio_output_root, "**/*.js")) | ||||||
|     .pipe(zopfli()) |     .pipe(zopfli(zopfliOptions)) | ||||||
|     .pipe(gulp.dest(paths.hassio_root)); |     .pipe(gulp.dest(paths.hassio_output_root)); | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -1,6 +1,8 @@ | |||||||
| // Run demo develop mode | // Run demo develop mode | ||||||
| const gulp = require("gulp"); | const gulp = require("gulp"); | ||||||
|  |  | ||||||
|  | const env = require("../env"); | ||||||
|  |  | ||||||
| require("./clean.js"); | require("./clean.js"); | ||||||
| require("./translations.js"); | require("./translations.js"); | ||||||
| require("./gen-icons-json.js"); | require("./gen-icons-json.js"); | ||||||
| @@ -8,6 +10,7 @@ require("./gather-static.js"); | |||||||
| require("./webpack.js"); | require("./webpack.js"); | ||||||
| require("./service-worker.js"); | require("./service-worker.js"); | ||||||
| require("./entry-html.js"); | require("./entry-html.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-demo", |   "develop-demo", | ||||||
| @@ -19,7 +22,7 @@ gulp.task( | |||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "gen-index-demo-dev", "build-translations"), |     gulp.parallel("gen-icons-json", "gen-index-demo-dev", "build-translations"), | ||||||
|     "copy-static-demo", |     "copy-static-demo", | ||||||
|     "webpack-dev-server-demo" |     env.useRollup() ? "rollup-dev-server-demo" : "webpack-dev-server-demo" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -34,7 +37,7 @@ gulp.task( | |||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations"), |     gulp.parallel("gen-icons-json", "build-translations"), | ||||||
|     "copy-static-demo", |     "copy-static-demo", | ||||||
|     "webpack-prod-demo", |     env.useRollup() ? "rollup-prod-demo" : "webpack-prod-demo", | ||||||
|     "gen-index-demo-prod" |     "gen-index-demo-prod" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -6,31 +6,36 @@ const fs = require("fs-extra"); | |||||||
| const path = require("path"); | const path = require("path"); | ||||||
| const template = require("lodash.template"); | const template = require("lodash.template"); | ||||||
| const minify = require("html-minifier").minify; | const minify = require("html-minifier").minify; | ||||||
| const config = require("../paths.js"); | const paths = require("../paths.js"); | ||||||
|  | const env = require("../env.js"); | ||||||
|  |  | ||||||
| const templatePath = (tpl) => | const templatePath = (tpl) => | ||||||
|   path.resolve(config.polymer_dir, "src/html/", `${tpl}.html.template`); |   path.resolve(paths.polymer_dir, "src/html/", `${tpl}.html.template`); | ||||||
|  |  | ||||||
| const readFile = (pth) => fs.readFileSync(pth).toString(); | const readFile = (pth) => fs.readFileSync(pth).toString(); | ||||||
|  |  | ||||||
| const renderTemplate = (pth, data = {}, pathFunc = templatePath) => { | const renderTemplate = (pth, data = {}, pathFunc = templatePath) => { | ||||||
|   const compiled = template(readFile(pathFunc(pth))); |   const compiled = template(readFile(pathFunc(pth))); | ||||||
|   return compiled({ ...data, renderTemplate }); |   return compiled({ | ||||||
|  |     ...data, | ||||||
|  |     useRollup: env.useRollup(), | ||||||
|  |     renderTemplate, | ||||||
|  |   }); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const renderDemoTemplate = (pth, data = {}) => | const renderDemoTemplate = (pth, data = {}) => | ||||||
|   renderTemplate(pth, data, (tpl) => |   renderTemplate(pth, data, (tpl) => | ||||||
|     path.resolve(config.demo_dir, "src/html/", `${tpl}.html.template`) |     path.resolve(paths.demo_dir, "src/html/", `${tpl}.html.template`) | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
| const renderCastTemplate = (pth, data = {}) => | const renderCastTemplate = (pth, data = {}) => | ||||||
|   renderTemplate(pth, data, (tpl) => |   renderTemplate(pth, data, (tpl) => | ||||||
|     path.resolve(config.cast_dir, "src/html/", `${tpl}.html.template`) |     path.resolve(paths.cast_dir, "src/html/", `${tpl}.html.template`) | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
| const renderGalleryTemplate = (pth, data = {}) => | const renderGalleryTemplate = (pth, data = {}) => | ||||||
|   renderTemplate(pth, data, (tpl) => |   renderTemplate(pth, data, (tpl) => | ||||||
|     path.resolve(config.gallery_dir, "src/html/", `${tpl}.html.template`) |     path.resolve(paths.gallery_dir, "src/html/", `${tpl}.html.template`) | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
| const minifyHtml = (content) => | const minifyHtml = (content) => | ||||||
| @@ -48,29 +53,36 @@ gulp.task("gen-pages-dev", (done) => { | |||||||
|     const content = renderTemplate(page, { |     const content = renderTemplate(page, { | ||||||
|       latestPageJS: `/frontend_latest/${page}.js`, |       latestPageJS: `/frontend_latest/${page}.js`, | ||||||
|  |  | ||||||
|       es5Compatibility: "/frontend_es5/compatibility.js", |  | ||||||
|       es5PageJS: `/frontend_es5/${page}.js`, |       es5PageJS: `/frontend_es5/${page}.js`, | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     fs.outputFileSync(path.resolve(config.root, `${page}.html`), content); |     fs.outputFileSync( | ||||||
|  |       path.resolve(paths.app_output_root, `${page}.html`), | ||||||
|  |       content | ||||||
|  |     ); | ||||||
|   } |   } | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-pages-prod", (done) => { | gulp.task("gen-pages-prod", (done) => { | ||||||
|   const latestManifest = require(path.resolve(config.output, "manifest.json")); |   const latestManifest = require(path.resolve( | ||||||
|   const es5Manifest = require(path.resolve(config.output_es5, "manifest.json")); |     paths.app_output_latest, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |   const es5Manifest = require(path.resolve( | ||||||
|  |     paths.app_output_es5, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |  | ||||||
|   for (const page of PAGES) { |   for (const page of PAGES) { | ||||||
|     const content = renderTemplate(page, { |     const content = renderTemplate(page, { | ||||||
|       latestPageJS: latestManifest[`${page}.js`], |       latestPageJS: latestManifest[`${page}.js`], | ||||||
|  |  | ||||||
|       es5Compatibility: es5Manifest["compatibility.js"], |  | ||||||
|       es5PageJS: es5Manifest[`${page}.js`], |       es5PageJS: es5Manifest[`${page}.js`], | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     fs.outputFileSync( |     fs.outputFileSync( | ||||||
|       path.resolve(config.root, `${page}.html`), |       path.resolve(paths.app_output_root, `${page}.html`), | ||||||
|       minifyHtml(content) |       minifyHtml(content) | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| @@ -78,39 +90,44 @@ gulp.task("gen-pages-prod", (done) => { | |||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-index-app-dev", (done) => { | gulp.task("gen-index-app-dev", (done) => { | ||||||
|   // In dev mode we don't mangle names, so we hardcode urls. That way we can |  | ||||||
|   // run webpack as last in watch mode, which blocks output. |  | ||||||
|   const content = renderTemplate("index", { |   const content = renderTemplate("index", { | ||||||
|     latestAppJS: "/frontend_latest/app.js", |     latestAppJS: "/frontend_latest/app.js", | ||||||
|     latestCoreJS: "/frontend_latest/core.js", |     latestCoreJS: "/frontend_latest/core.js", | ||||||
|     latestCustomPanelJS: "/frontend_latest/custom-panel.js", |     latestCustomPanelJS: "/frontend_latest/custom-panel.js", | ||||||
|  |  | ||||||
|     es5Compatibility: "/frontend_es5/compatibility.js", |  | ||||||
|     es5AppJS: "/frontend_es5/app.js", |     es5AppJS: "/frontend_es5/app.js", | ||||||
|     es5CoreJS: "/frontend_es5/core.js", |     es5CoreJS: "/frontend_es5/core.js", | ||||||
|     es5CustomPanelJS: "/frontend_es5/custom-panel.js", |     es5CustomPanelJS: "/frontend_es5/custom-panel.js", | ||||||
|   }).replace(/#THEMEC/g, "{{ theme_color }}"); |   }).replace(/#THEMEC/g, "{{ theme_color }}"); | ||||||
|  |  | ||||||
|   fs.outputFileSync(path.resolve(config.root, "index.html"), content); |   fs.outputFileSync(path.resolve(paths.app_output_root, "index.html"), content); | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-index-app-prod", (done) => { | gulp.task("gen-index-app-prod", (done) => { | ||||||
|   const latestManifest = require(path.resolve(config.output, "manifest.json")); |   const latestManifest = require(path.resolve( | ||||||
|   const es5Manifest = require(path.resolve(config.output_es5, "manifest.json")); |     paths.app_output_latest, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |   const es5Manifest = require(path.resolve( | ||||||
|  |     paths.app_output_es5, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|   const content = renderTemplate("index", { |   const content = renderTemplate("index", { | ||||||
|     latestAppJS: latestManifest["app.js"], |     latestAppJS: latestManifest["app.js"], | ||||||
|     latestCoreJS: latestManifest["core.js"], |     latestCoreJS: latestManifest["core.js"], | ||||||
|     latestCustomPanelJS: latestManifest["custom-panel.js"], |     latestCustomPanelJS: latestManifest["custom-panel.js"], | ||||||
|  |  | ||||||
|     es5Compatibility: es5Manifest["compatibility.js"], |  | ||||||
|     es5AppJS: es5Manifest["app.js"], |     es5AppJS: es5Manifest["app.js"], | ||||||
|     es5CoreJS: es5Manifest["core.js"], |     es5CoreJS: es5Manifest["core.js"], | ||||||
|     es5CustomPanelJS: es5Manifest["custom-panel.js"], |     es5CustomPanelJS: es5Manifest["custom-panel.js"], | ||||||
|   }); |   }); | ||||||
|   const minified = minifyHtml(content).replace(/#THEMEC/g, "{{ theme_color }}"); |   const minified = minifyHtml(content).replace(/#THEMEC/g, "{{ theme_color }}"); | ||||||
|  |  | ||||||
|   fs.outputFileSync(path.resolve(config.root, "index.html"), minified); |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.app_output_root, "index.html"), | ||||||
|  |     minified | ||||||
|  |   ); | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| @@ -119,7 +136,7 @@ gulp.task("gen-index-cast-dev", (done) => { | |||||||
|     latestReceiverJS: "/frontend_latest/receiver.js", |     latestReceiverJS: "/frontend_latest/receiver.js", | ||||||
|   }); |   }); | ||||||
|   fs.outputFileSync( |   fs.outputFileSync( | ||||||
|     path.resolve(config.cast_root, "receiver.html"), |     path.resolve(paths.cast_output_root, "receiver.html"), | ||||||
|     contentReceiver |     contentReceiver | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
| @@ -127,14 +144,17 @@ gulp.task("gen-index-cast-dev", (done) => { | |||||||
|     latestLauncherJS: "/frontend_latest/launcher.js", |     latestLauncherJS: "/frontend_latest/launcher.js", | ||||||
|     es5LauncherJS: "/frontend_es5/launcher.js", |     es5LauncherJS: "/frontend_es5/launcher.js", | ||||||
|   }); |   }); | ||||||
|   fs.outputFileSync(path.resolve(config.cast_root, "faq.html"), contentFAQ); |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.cast_output_root, "faq.html"), | ||||||
|  |     contentFAQ | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   const contentLauncher = renderCastTemplate("launcher", { |   const contentLauncher = renderCastTemplate("launcher", { | ||||||
|     latestLauncherJS: "/frontend_latest/launcher.js", |     latestLauncherJS: "/frontend_latest/launcher.js", | ||||||
|     es5LauncherJS: "/frontend_es5/launcher.js", |     es5LauncherJS: "/frontend_es5/launcher.js", | ||||||
|   }); |   }); | ||||||
|   fs.outputFileSync( |   fs.outputFileSync( | ||||||
|     path.resolve(config.cast_root, "index.html"), |     path.resolve(paths.cast_output_root, "index.html"), | ||||||
|     contentLauncher |     contentLauncher | ||||||
|   ); |   ); | ||||||
|   done(); |   done(); | ||||||
| @@ -142,11 +162,11 @@ gulp.task("gen-index-cast-dev", (done) => { | |||||||
|  |  | ||||||
| gulp.task("gen-index-cast-prod", (done) => { | gulp.task("gen-index-cast-prod", (done) => { | ||||||
|   const latestManifest = require(path.resolve( |   const latestManifest = require(path.resolve( | ||||||
|     config.cast_output, |     paths.cast_output_latest, | ||||||
|     "manifest.json" |     "manifest.json" | ||||||
|   )); |   )); | ||||||
|   const es5Manifest = require(path.resolve( |   const es5Manifest = require(path.resolve( | ||||||
|     config.cast_output_es5, |     paths.cast_output_es5, | ||||||
|     "manifest.json" |     "manifest.json" | ||||||
|   )); |   )); | ||||||
|  |  | ||||||
| @@ -154,7 +174,7 @@ gulp.task("gen-index-cast-prod", (done) => { | |||||||
|     latestReceiverJS: latestManifest["receiver.js"], |     latestReceiverJS: latestManifest["receiver.js"], | ||||||
|   }); |   }); | ||||||
|   fs.outputFileSync( |   fs.outputFileSync( | ||||||
|     path.resolve(config.cast_root, "receiver.html"), |     path.resolve(paths.cast_output_root, "receiver.html"), | ||||||
|     contentReceiver |     contentReceiver | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
| @@ -162,68 +182,74 @@ gulp.task("gen-index-cast-prod", (done) => { | |||||||
|     latestLauncherJS: latestManifest["launcher.js"], |     latestLauncherJS: latestManifest["launcher.js"], | ||||||
|     es5LauncherJS: es5Manifest["launcher.js"], |     es5LauncherJS: es5Manifest["launcher.js"], | ||||||
|   }); |   }); | ||||||
|   fs.outputFileSync(path.resolve(config.cast_root, "faq.html"), contentFAQ); |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.cast_output_root, "faq.html"), | ||||||
|  |     contentFAQ | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   const contentLauncher = renderCastTemplate("launcher", { |   const contentLauncher = renderCastTemplate("launcher", { | ||||||
|     latestLauncherJS: latestManifest["launcher.js"], |     latestLauncherJS: latestManifest["launcher.js"], | ||||||
|     es5LauncherJS: es5Manifest["launcher.js"], |     es5LauncherJS: es5Manifest["launcher.js"], | ||||||
|   }); |   }); | ||||||
|   fs.outputFileSync( |   fs.outputFileSync( | ||||||
|     path.resolve(config.cast_root, "index.html"), |     path.resolve(paths.cast_output_root, "index.html"), | ||||||
|     contentLauncher |     contentLauncher | ||||||
|   ); |   ); | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-index-demo-dev", (done) => { | gulp.task("gen-index-demo-dev", (done) => { | ||||||
|   // In dev mode we don't mangle names, so we hardcode urls. That way we can |  | ||||||
|   // run webpack as last in watch mode, which blocks output. |  | ||||||
|   const content = renderDemoTemplate("index", { |   const content = renderDemoTemplate("index", { | ||||||
|     latestDemoJS: "/frontend_latest/main.js", |     latestDemoJS: "/frontend_latest/main.js", | ||||||
|  |  | ||||||
|     es5Compatibility: "/frontend_es5/compatibility.js", |  | ||||||
|     es5DemoJS: "/frontend_es5/main.js", |     es5DemoJS: "/frontend_es5/main.js", | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   fs.outputFileSync(path.resolve(config.demo_root, "index.html"), content); |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.demo_output_root, "index.html"), | ||||||
|  |     content | ||||||
|  |   ); | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-index-demo-prod", (done) => { | gulp.task("gen-index-demo-prod", (done) => { | ||||||
|   const latestManifest = require(path.resolve( |   const latestManifest = require(path.resolve( | ||||||
|     config.demo_output, |     paths.demo_output_latest, | ||||||
|     "manifest.json" |     "manifest.json" | ||||||
|   )); |   )); | ||||||
|   const es5Manifest = require(path.resolve( |   const es5Manifest = require(path.resolve( | ||||||
|     config.demo_output_es5, |     paths.demo_output_es5, | ||||||
|     "manifest.json" |     "manifest.json" | ||||||
|   )); |   )); | ||||||
|   const content = renderDemoTemplate("index", { |   const content = renderDemoTemplate("index", { | ||||||
|     latestDemoJS: latestManifest["main.js"], |     latestDemoJS: latestManifest["main.js"], | ||||||
|  |  | ||||||
|     es5Compatibility: es5Manifest["compatibility.js"], |  | ||||||
|     es5DemoJS: es5Manifest["main.js"], |     es5DemoJS: es5Manifest["main.js"], | ||||||
|   }); |   }); | ||||||
|   const minified = minifyHtml(content); |   const minified = minifyHtml(content); | ||||||
|  |  | ||||||
|   fs.outputFileSync(path.resolve(config.demo_root, "index.html"), minified); |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.demo_output_root, "index.html"), | ||||||
|  |     minified | ||||||
|  |   ); | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-index-gallery-dev", (done) => { | gulp.task("gen-index-gallery-dev", (done) => { | ||||||
|   // In dev mode we don't mangle names, so we hardcode urls. That way we can |  | ||||||
|   // run webpack as last in watch mode, which blocks output. |  | ||||||
|   const content = renderGalleryTemplate("index", { |   const content = renderGalleryTemplate("index", { | ||||||
|     latestGalleryJS: "./frontend_latest/entrypoint.js", |     latestGalleryJS: "./frontend_latest/entrypoint.js", | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   fs.outputFileSync(path.resolve(config.gallery_root, "index.html"), content); |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.gallery_output_root, "index.html"), | ||||||
|  |     content | ||||||
|  |   ); | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("gen-index-gallery-prod", (done) => { | gulp.task("gen-index-gallery-prod", (done) => { | ||||||
|   const latestManifest = require(path.resolve( |   const latestManifest = require(path.resolve( | ||||||
|     config.gallery_output, |     paths.gallery_output_latest, | ||||||
|     "manifest.json" |     "manifest.json" | ||||||
|   )); |   )); | ||||||
|   const content = renderGalleryTemplate("index", { |   const content = renderGalleryTemplate("index", { | ||||||
| @@ -231,6 +257,48 @@ gulp.task("gen-index-gallery-prod", (done) => { | |||||||
|   }); |   }); | ||||||
|   const minified = minifyHtml(content); |   const minified = minifyHtml(content); | ||||||
|  |  | ||||||
|   fs.outputFileSync(path.resolve(config.gallery_root, "index.html"), minified); |   fs.outputFileSync( | ||||||
|  |     path.resolve(paths.gallery_output_root, "index.html"), | ||||||
|  |     minified | ||||||
|  |   ); | ||||||
|   done(); |   done(); | ||||||
| }); | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-hassio-dev", async () => { | ||||||
|  |   writeHassioEntrypoint( | ||||||
|  |     `${paths.hassio_publicPath}/frontend_latest/entrypoint.js`, | ||||||
|  |     `${paths.hassio_publicPath}/frontend_es5/entrypoint.js` | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("gen-index-hassio-prod", async () => { | ||||||
|  |   const latestManifest = require(path.resolve( | ||||||
|  |     paths.hassio_output_latest, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |   const es5Manifest = require(path.resolve( | ||||||
|  |     paths.hassio_output_es5, | ||||||
|  |     "manifest.json" | ||||||
|  |   )); | ||||||
|  |   writeHassioEntrypoint( | ||||||
|  |     latestManifest["entrypoint.js"], | ||||||
|  |     es5Manifest["entrypoint.js"] | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | function writeHassioEntrypoint(latestEntrypoint, es5Entrypoint) { | ||||||
|  |   fs.mkdirSync(paths.hassio_output_root, { recursive: true }); | ||||||
|  |   fs.writeFileSync( | ||||||
|  |     path.resolve(paths.hassio_output_root, "entrypoint.js"), | ||||||
|  |     ` | ||||||
|  | try { | ||||||
|  |   new Function("import('${latestEntrypoint}')")(); | ||||||
|  | } catch (err) { | ||||||
|  |   var el = document.createElement('script'); | ||||||
|  |   el.src = '${es5Entrypoint}'; | ||||||
|  |   document.body.appendChild(el); | ||||||
|  | } | ||||||
|  |   `, | ||||||
|  |     { encoding: "utf-8" } | ||||||
|  |   ); | ||||||
|  | } | ||||||
|   | |||||||
| @@ -1,5 +1,10 @@ | |||||||
| // Run demo develop mode | // Run demo develop mode | ||||||
| const gulp = require("gulp"); | const gulp = require("gulp"); | ||||||
|  | const fs = require("fs"); | ||||||
|  | const path = require("path"); | ||||||
|  |  | ||||||
|  | const env = require("../env"); | ||||||
|  | const paths = require("../paths"); | ||||||
|  |  | ||||||
| require("./clean.js"); | require("./clean.js"); | ||||||
| require("./translations.js"); | require("./translations.js"); | ||||||
| @@ -8,6 +13,32 @@ require("./gather-static.js"); | |||||||
| require("./webpack.js"); | require("./webpack.js"); | ||||||
| require("./service-worker.js"); | require("./service-worker.js"); | ||||||
| require("./entry-html.js"); | require("./entry-html.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  |  | ||||||
|  | gulp.task("gather-gallery-demos", async function gatherDemos() { | ||||||
|  |   const files = await fs.promises.readdir( | ||||||
|  |     path.resolve(paths.gallery_dir, "src/demos") | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   let content = "export const DEMOS = {\n"; | ||||||
|  |  | ||||||
|  |   for (const file of files) { | ||||||
|  |     const demoId = path.basename(file, ".ts"); | ||||||
|  |     const demoPath = "../src/demos/" + demoId; | ||||||
|  |     content += `  "${demoId}": () => import("${demoPath}"),\n`; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   content += "};"; | ||||||
|  |  | ||||||
|  |   const galleryBuild = path.resolve(paths.gallery_dir, "build"); | ||||||
|  |  | ||||||
|  |   fs.mkdirSync(galleryBuild, { recursive: true }); | ||||||
|  |   fs.writeFileSync( | ||||||
|  |     path.resolve(galleryBuild, "import-demos.ts"), | ||||||
|  |     content, | ||||||
|  |     "utf-8" | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-gallery", |   "develop-gallery", | ||||||
| @@ -17,10 +48,14 @@ gulp.task( | |||||||
|     }, |     }, | ||||||
|     "clean-gallery", |     "clean-gallery", | ||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations"), |     gulp.parallel( | ||||||
|  |       "gen-icons-json", | ||||||
|  |       "build-translations", | ||||||
|  |       "gather-gallery-demos" | ||||||
|  |     ), | ||||||
|     "copy-static-gallery", |     "copy-static-gallery", | ||||||
|     "gen-index-gallery-dev", |     "gen-index-gallery-dev", | ||||||
|     "webpack-dev-server-gallery" |     env.useRollup() ? "rollup-dev-server-gallery" : "webpack-dev-server-gallery" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -32,9 +67,13 @@ gulp.task( | |||||||
|     }, |     }, | ||||||
|     "clean-gallery", |     "clean-gallery", | ||||||
|     "translations-enable-merge-backend", |     "translations-enable-merge-backend", | ||||||
|     gulp.parallel("gen-icons-json", "build-translations"), |     gulp.parallel( | ||||||
|  |       "gen-icons-json", | ||||||
|  |       "build-translations", | ||||||
|  |       "gather-gallery-demos" | ||||||
|  |     ), | ||||||
|     "copy-static-gallery", |     "copy-static-gallery", | ||||||
|     "webpack-prod-gallery", |     env.useRollup() ? "rollup-prod-gallery" : "webpack-prod-gallery", | ||||||
|     "gen-index-gallery-prod" |     "gen-index-gallery-prod" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
| @@ -36,11 +36,13 @@ function copyMdiIcons(staticDir) { | |||||||
| function copyPolyfills(staticDir) { | function copyPolyfills(staticDir) { | ||||||
|   const staticPath = genStaticPath(staticDir); |   const staticPath = genStaticPath(staticDir); | ||||||
|  |  | ||||||
|   // Web Component polyfills and adapters |   // For custom panels using ES5 builds that don't use Babel 7+ | ||||||
|   copyFileDir( |   copyFileDir( | ||||||
|     npmPath("@webcomponents/webcomponentsjs/custom-elements-es5-adapter.js"), |     npmPath("@webcomponents/webcomponentsjs/custom-elements-es5-adapter.js"), | ||||||
|     staticPath("polyfills/") |     staticPath("polyfills/") | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|  |   // Web Component polyfills and adapters | ||||||
|   copyFileDir( |   copyFileDir( | ||||||
|     npmPath("@webcomponents/webcomponentsjs/webcomponents-bundle.js"), |     npmPath("@webcomponents/webcomponentsjs/webcomponents-bundle.js"), | ||||||
|     staticPath("polyfills/") |     staticPath("polyfills/") | ||||||
| @@ -51,6 +53,12 @@ function copyPolyfills(staticDir) { | |||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | function copyLoaderJS(staticDir) { | ||||||
|  |   const staticPath = genStaticPath(staticDir); | ||||||
|  |   copyFileDir(npmPath("systemjs/dist/s.min.js"), staticPath("js")); | ||||||
|  |   copyFileDir(npmPath("systemjs/dist/s.min.js.map"), staticPath("js")); | ||||||
|  | } | ||||||
|  |  | ||||||
| function copyFonts(staticDir) { | function copyFonts(staticDir) { | ||||||
|   const staticPath = genStaticPath(staticDir); |   const staticPath = genStaticPath(staticDir); | ||||||
|   // Local fonts |   // Local fonts | ||||||
| @@ -72,17 +80,17 @@ function copyMapPanel(staticDir) { | |||||||
|   ); |   ); | ||||||
| } | } | ||||||
|  |  | ||||||
| gulp.task("copy-translations", (done) => { | gulp.task("copy-translations-app", async () => { | ||||||
|   const staticDir = paths.static; |   const staticDir = paths.app_output_static; | ||||||
|   copyTranslations(staticDir); |   copyTranslations(staticDir); | ||||||
|   done(); |  | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("copy-static", (done) => { | gulp.task("copy-static-app", async () => { | ||||||
|   const staticDir = paths.static; |   const staticDir = paths.app_output_static; | ||||||
|   // Basic static files |   // Basic static files | ||||||
|   fs.copySync(polyPath("public"), paths.root); |   fs.copySync(polyPath("public"), paths.app_output_root); | ||||||
|  |  | ||||||
|  |   copyLoaderJS(staticDir); | ||||||
|   copyPolyfills(staticDir); |   copyPolyfills(staticDir); | ||||||
|   copyFonts(staticDir); |   copyFonts(staticDir); | ||||||
|   copyTranslations(staticDir); |   copyTranslations(staticDir); | ||||||
| @@ -90,48 +98,50 @@ gulp.task("copy-static", (done) => { | |||||||
|  |  | ||||||
|   // Panel assets |   // Panel assets | ||||||
|   copyMapPanel(staticDir); |   copyMapPanel(staticDir); | ||||||
|   done(); |  | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("copy-static-demo", (done) => { | gulp.task("copy-static-demo", async () => { | ||||||
|   // Copy app static files |   // Copy app static files | ||||||
|   fs.copySync( |   fs.copySync( | ||||||
|     polyPath("public/static"), |     polyPath("public/static"), | ||||||
|     path.resolve(paths.demo_root, "static") |     path.resolve(paths.demo_output_root, "static") | ||||||
|   ); |   ); | ||||||
|   // Copy demo static files |   // Copy demo static files | ||||||
|   fs.copySync(path.resolve(paths.demo_dir, "public"), paths.demo_root); |   fs.copySync(path.resolve(paths.demo_dir, "public"), paths.demo_output_root); | ||||||
|  |  | ||||||
|   copyPolyfills(paths.demo_static); |   copyLoaderJS(paths.demo_output_static); | ||||||
|   copyMapPanel(paths.demo_static); |   copyPolyfills(paths.demo_output_static); | ||||||
|   copyFonts(paths.demo_static); |   copyMapPanel(paths.demo_output_static); | ||||||
|   copyTranslations(paths.demo_static); |   copyFonts(paths.demo_output_static); | ||||||
|   copyMdiIcons(paths.demo_static); |   copyTranslations(paths.demo_output_static); | ||||||
|   done(); |   copyMdiIcons(paths.demo_output_static); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("copy-static-cast", (done) => { | gulp.task("copy-static-cast", async () => { | ||||||
|   // Copy app static files |   // Copy app static files | ||||||
|   fs.copySync(polyPath("public/static"), paths.cast_static); |   fs.copySync(polyPath("public/static"), paths.cast_output_static); | ||||||
|   // Copy cast static files |   // Copy cast static files | ||||||
|   fs.copySync(path.resolve(paths.cast_dir, "public"), paths.cast_root); |   fs.copySync(path.resolve(paths.cast_dir, "public"), paths.cast_output_root); | ||||||
|  |  | ||||||
|   copyMapPanel(paths.cast_static); |   copyLoaderJS(paths.cast_output_static); | ||||||
|   copyFonts(paths.cast_static); |   copyPolyfills(paths.cast_output_static); | ||||||
|   copyTranslations(paths.cast_static); |   copyMapPanel(paths.cast_output_static); | ||||||
|   copyMdiIcons(paths.cast_static); |   copyFonts(paths.cast_output_static); | ||||||
|   done(); |   copyTranslations(paths.cast_output_static); | ||||||
|  |   copyMdiIcons(paths.cast_output_static); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| gulp.task("copy-static-gallery", (done) => { | gulp.task("copy-static-gallery", async () => { | ||||||
|   // Copy app static files |   // Copy app static files | ||||||
|   fs.copySync(polyPath("public/static"), paths.gallery_static); |   fs.copySync(polyPath("public/static"), paths.gallery_output_static); | ||||||
|   // Copy gallery static files |   // Copy gallery static files | ||||||
|   fs.copySync(path.resolve(paths.gallery_dir, "public"), paths.gallery_root); |   fs.copySync( | ||||||
|  |     path.resolve(paths.gallery_dir, "public"), | ||||||
|  |     paths.gallery_output_root | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   copyMapPanel(paths.gallery_static); |   copyMapPanel(paths.gallery_output_static); | ||||||
|   copyFonts(paths.gallery_static); |   copyFonts(paths.gallery_output_static); | ||||||
|   copyTranslations(paths.gallery_static); |   copyTranslations(paths.gallery_output_static); | ||||||
|   copyMdiIcons(paths.gallery_static); |   copyMdiIcons(paths.gallery_output_static); | ||||||
|   done(); |  | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ const META_PATH = path.resolve(ICON_PACKAGE_PATH, "meta.json"); | |||||||
| const PACKAGE_PATH = path.resolve(ICON_PACKAGE_PATH, "package.json"); | const PACKAGE_PATH = path.resolve(ICON_PACKAGE_PATH, "package.json"); | ||||||
| const ICON_PATH = path.resolve(ICON_PACKAGE_PATH, "svg"); | const ICON_PATH = path.resolve(ICON_PACKAGE_PATH, "svg"); | ||||||
| const OUTPUT_DIR = path.resolve(__dirname, "../../build/mdi"); | const OUTPUT_DIR = path.resolve(__dirname, "../../build/mdi"); | ||||||
|  | const REMOVED_ICONS_PATH = path.resolve(__dirname, "../removedIcons.json"); | ||||||
|  |  | ||||||
| const encoding = "utf8"; | const encoding = "utf8"; | ||||||
|  |  | ||||||
| @@ -25,6 +26,13 @@ const getMeta = () => { | |||||||
|   }); |   }); | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | const addRemovedMeta = (meta) => { | ||||||
|  |   const file = fs.readFileSync(REMOVED_ICONS_PATH, { encoding }); | ||||||
|  |   const removed = JSON.parse(file); | ||||||
|  |   const combinedMeta = [...meta, ...removed]; | ||||||
|  |   return combinedMeta.sort((a, b) => a.name.localeCompare(b.name)); | ||||||
|  | }; | ||||||
|  |  | ||||||
| const splitBySize = (meta) => { | const splitBySize = (meta) => { | ||||||
|   const chunks = []; |   const chunks = []; | ||||||
|   const CHUNK_SIZE = 50000; |   const CHUNK_SIZE = 50000; | ||||||
| @@ -69,7 +77,7 @@ const findDifferentiator = (curString, prevString) => { | |||||||
| }; | }; | ||||||
|  |  | ||||||
| gulp.task("gen-icons-json", (done) => { | gulp.task("gen-icons-json", (done) => { | ||||||
|   const meta = getMeta(); |   const meta = addRemovedMeta(getMeta()); | ||||||
|   const split = splitBySize(meta); |   const split = splitBySize(meta); | ||||||
|  |  | ||||||
|   if (!fs.existsSync(OUTPUT_DIR)) { |   if (!fs.existsSync(OUTPUT_DIR)) { | ||||||
|   | |||||||
| @@ -1,11 +1,15 @@ | |||||||
| const gulp = require("gulp"); | const gulp = require("gulp"); | ||||||
|  | const fs = require("fs"); | ||||||
|  | const path = require("path"); | ||||||
|  |  | ||||||
| const envVars = require("../env"); | const env = require("../env"); | ||||||
|  | const paths = require("../paths"); | ||||||
|  |  | ||||||
| require("./clean.js"); | require("./clean.js"); | ||||||
| require("./gen-icons-json.js"); | require("./gen-icons-json.js"); | ||||||
| require("./webpack.js"); | require("./webpack.js"); | ||||||
| require("./compress.js"); | require("./compress.js"); | ||||||
|  | require("./rollup.js"); | ||||||
|  |  | ||||||
| gulp.task( | gulp.task( | ||||||
|   "develop-hassio", |   "develop-hassio", | ||||||
| @@ -15,7 +19,8 @@ gulp.task( | |||||||
|     }, |     }, | ||||||
|     "clean-hassio", |     "clean-hassio", | ||||||
|     "gen-icons-json", |     "gen-icons-json", | ||||||
|     "webpack-watch-hassio" |     "gen-index-hassio-dev", | ||||||
|  |     env.useRollup() ? "rollup-watch-hassio" : "webpack-watch-hassio" | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|  |  | ||||||
| @@ -27,8 +32,9 @@ gulp.task( | |||||||
|     }, |     }, | ||||||
|     "clean-hassio", |     "clean-hassio", | ||||||
|     "gen-icons-json", |     "gen-icons-json", | ||||||
|     "webpack-prod-hassio", |     env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio", | ||||||
|  |     "gen-index-hassio-prod", | ||||||
|     ...// Don't compress running tests |     ...// Don't compress running tests | ||||||
|     (envVars.isTest() ? [] : ["compress-hassio"]) |     (env.isTest() ? [] : ["compress-hassio"]) | ||||||
|   ) |   ) | ||||||
| ); | ); | ||||||
|   | |||||||
							
								
								
									
										146
									
								
								build-scripts/gulp/rollup.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,146 @@ | |||||||
|  | // Tasks to run Rollup | ||||||
|  | const path = require("path"); | ||||||
|  | const gulp = require("gulp"); | ||||||
|  | const rollup = require("rollup"); | ||||||
|  | const handler = require("serve-handler"); | ||||||
|  | const http = require("http"); | ||||||
|  | const log = require("fancy-log"); | ||||||
|  | const rollupConfig = require("../rollup"); | ||||||
|  | const paths = require("../paths"); | ||||||
|  | const open = require("open"); | ||||||
|  |  | ||||||
|  | const bothBuilds = (createConfigFunc, params) => | ||||||
|  |   gulp.series( | ||||||
|  |     async function buildLatest() { | ||||||
|  |       await buildRollup( | ||||||
|  |         createConfigFunc({ | ||||||
|  |           ...params, | ||||||
|  |           latestBuild: true, | ||||||
|  |         }) | ||||||
|  |       ); | ||||||
|  |     }, | ||||||
|  |     async function buildES5() { | ||||||
|  |       await buildRollup( | ||||||
|  |         createConfigFunc({ | ||||||
|  |           ...params, | ||||||
|  |           latestBuild: false, | ||||||
|  |         }) | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  | function createServer(serveOptions) { | ||||||
|  |   const server = http.createServer((request, response) => { | ||||||
|  |     return handler(request, response, { | ||||||
|  |       public: serveOptions.root, | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   server.listen( | ||||||
|  |     serveOptions.port, | ||||||
|  |     serveOptions.networkAccess ? "0.0.0.0" : undefined, | ||||||
|  |     () => { | ||||||
|  |       log.info(`Available at http://localhost:${serveOptions.port}`); | ||||||
|  |       open(`http://localhost:${serveOptions.port}`); | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function watchRollup(createConfig, extraWatchSrc = [], serveOptions) { | ||||||
|  |   const { inputOptions, outputOptions } = createConfig({ | ||||||
|  |     isProdBuild: false, | ||||||
|  |     latestBuild: true, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   const watcher = rollup.watch({ | ||||||
|  |     ...inputOptions, | ||||||
|  |     output: [outputOptions], | ||||||
|  |     watch: { | ||||||
|  |       include: ["src/**"] + extraWatchSrc, | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   let startedHttp = false; | ||||||
|  |  | ||||||
|  |   watcher.on("event", (event) => { | ||||||
|  |     if (event.code === "BUNDLE_END") { | ||||||
|  |       log(`Build done @ ${new Date().toLocaleTimeString()}`); | ||||||
|  |     } else if (event.code === "ERROR") { | ||||||
|  |       log.error(event.error); | ||||||
|  |     } else if (event.code === "END") { | ||||||
|  |       if (startedHttp || !serveOptions) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       startedHttp = true; | ||||||
|  |       createServer(serveOptions); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   gulp.watch( | ||||||
|  |     path.join(paths.translations_src, "en.json"), | ||||||
|  |     gulp.series("build-translations", "copy-translations-app") | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | async function buildRollup(config) { | ||||||
|  |   const bundle = await rollup.rollup(config.inputOptions); | ||||||
|  |   await bundle.write(config.outputOptions); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | gulp.task("rollup-watch-app", () => { | ||||||
|  |   watchRollup(rollupConfig.createAppConfig); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("rollup-watch-hassio", () => { | ||||||
|  |   watchRollup(rollupConfig.createHassioConfig, ["hassio/src/**"]); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("rollup-dev-server-demo", () => { | ||||||
|  |   watchRollup(rollupConfig.createDemoConfig, ["demo/src/**"], { | ||||||
|  |     root: paths.demo_output_root, | ||||||
|  |     port: 8090, | ||||||
|  |   }); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("rollup-dev-server-cast", () => { | ||||||
|  |   watchRollup(rollupConfig.createCastConfig, ["cast/src/**"], { | ||||||
|  |     root: paths.cast_output_root, | ||||||
|  |     port: 8080, | ||||||
|  |     networkAccess: true, | ||||||
|  |   }); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task("rollup-dev-server-gallery", () => { | ||||||
|  |   watchRollup(rollupConfig.createGalleryConfig, ["gallery/src/**"], { | ||||||
|  |     root: paths.gallery_output_root, | ||||||
|  |     port: 8100, | ||||||
|  |   }); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | gulp.task( | ||||||
|  |   "rollup-prod-app", | ||||||
|  |   bothBuilds(rollupConfig.createAppConfig, { isProdBuild: true }) | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | gulp.task( | ||||||
|  |   "rollup-prod-demo", | ||||||
|  |   bothBuilds(rollupConfig.createDemoConfig, { isProdBuild: true }) | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | gulp.task( | ||||||
|  |   "rollup-prod-cast", | ||||||
|  |   bothBuilds(rollupConfig.createCastConfig, { isProdBuild: true }) | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | gulp.task("rollup-prod-hassio", () => | ||||||
|  |   bothBuilds(rollupConfig.createHassioConfig, { isProdBuild: true }) | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | gulp.task("rollup-prod-gallery", () => | ||||||
|  |   buildRollup( | ||||||
|  |     rollupConfig.createGalleryConfig({ | ||||||
|  |       isProdBuild: true, | ||||||
|  |       latestBuild: true, | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  | ); | ||||||
| @@ -9,7 +9,7 @@ const workboxBuild = require("workbox-build"); | |||||||
| const sourceMapUrl = require("source-map-url"); | const sourceMapUrl = require("source-map-url"); | ||||||
| const paths = require("../paths.js"); | const paths = require("../paths.js"); | ||||||
|  |  | ||||||
| const swDest = path.resolve(paths.root, "service_worker.js"); | const swDest = path.resolve(paths.app_output_root, "service_worker.js"); | ||||||
|  |  | ||||||
| const writeSW = (content) => fs.outputFileSync(swDest, content.trim() + "\n"); | const writeSW = (content) => fs.outputFileSync(swDest, content.trim() + "\n"); | ||||||
|  |  | ||||||
| @@ -31,32 +31,38 @@ self.addEventListener('install', (event) => { | |||||||
| gulp.task("gen-service-worker-app-prod", async () => { | gulp.task("gen-service-worker-app-prod", async () => { | ||||||
|   // Read bundled source file |   // Read bundled source file | ||||||
|   const bundleManifestLatest = require(path.resolve( |   const bundleManifestLatest = require(path.resolve( | ||||||
|     paths.output, |     paths.app_output_latest, | ||||||
|     "manifest.json" |     "manifest.json" | ||||||
|   )); |   )); | ||||||
|   let serviceWorkerContent = fs.readFileSync( |   let serviceWorkerContent = fs.readFileSync( | ||||||
|     paths.root + bundleManifestLatest["service_worker.js"], |     paths.app_output_root + bundleManifestLatest["service_worker.js"], | ||||||
|     "utf-8" |     "utf-8" | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   // Delete old file from frontend_latest so manifest won't pick it up |   // Delete old file from frontend_latest so manifest won't pick it up | ||||||
|   fs.removeSync(paths.root + bundleManifestLatest["service_worker.js"]); |   fs.removeSync( | ||||||
|   fs.removeSync(paths.root + bundleManifestLatest["service_worker.js.map"]); |     paths.app_output_root + bundleManifestLatest["service_worker.js"] | ||||||
|  |   ); | ||||||
|  |   fs.removeSync( | ||||||
|  |     paths.app_output_root + bundleManifestLatest["service_worker.js.map"] | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   // Remove ES5 |   // Remove ES5 | ||||||
|   const bundleManifestES5 = require(path.resolve( |   const bundleManifestES5 = require(path.resolve( | ||||||
|     paths.output_es5, |     paths.app_output_es5, | ||||||
|     "manifest.json" |     "manifest.json" | ||||||
|   )); |   )); | ||||||
|   fs.removeSync(paths.root + bundleManifestES5["service_worker.js"]); |   fs.removeSync(paths.app_output_root + bundleManifestES5["service_worker.js"]); | ||||||
|   fs.removeSync(paths.root + bundleManifestES5["service_worker.js.map"]); |   fs.removeSync( | ||||||
|  |     paths.app_output_root + bundleManifestES5["service_worker.js.map"] | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   const workboxManifest = await workboxBuild.getManifest({ |   const workboxManifest = await workboxBuild.getManifest({ | ||||||
|     // Files that mach this pattern will be considered unique and skip revision check |     // Files that mach this pattern will be considered unique and skip revision check | ||||||
|     // ignore JS files + translation files |     // ignore JS files + translation files | ||||||
|     dontCacheBustURLsMatching: /(frontend_latest\/.+|static\/translations\/.+)/, |     dontCacheBustURLsMatching: /(frontend_latest\/.+|static\/translations\/.+)/, | ||||||
|  |  | ||||||
|     globDirectory: paths.root, |     globDirectory: paths.app_output_root, | ||||||
|     globPatterns: [ |     globPatterns: [ | ||||||
|       "frontend_latest/*.js", |       "frontend_latest/*.js", | ||||||
|       // Cache all English translations because we catch them as fallback |       // Cache all English translations because we catch them as fallback | ||||||
|   | |||||||
| @@ -7,7 +7,6 @@ const gulp = require("gulp"); | |||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const foreach = require("gulp-foreach"); | const foreach = require("gulp-foreach"); | ||||||
| const merge = require("gulp-merge-json"); | const merge = require("gulp-merge-json"); | ||||||
| const minify = require("gulp-jsonminify"); |  | ||||||
| const rename = require("gulp-rename"); | const rename = require("gulp-rename"); | ||||||
| const transform = require("gulp-json-transform"); | const transform = require("gulp-json-transform"); | ||||||
| const { mapFiles } = require("../util"); | const { mapFiles } = require("../util"); | ||||||
| @@ -301,7 +300,6 @@ gulp.task("build-flattened-translations", function () { | |||||||
|         return flatten(data); |         return flatten(data); | ||||||
|       }) |       }) | ||||||
|     ) |     ) | ||||||
|     .pipe(minify()) |  | ||||||
|     .pipe( |     .pipe( | ||||||
|       rename((filePath) => { |       rename((filePath) => { | ||||||
|         if (filePath.dirname === "core") { |         if (filePath.dirname === "core") { | ||||||
|   | |||||||
| @@ -18,6 +18,14 @@ const bothBuilds = (createConfigFunc, params) => [ | |||||||
|   createConfigFunc({ ...params, latestBuild: false }), |   createConfigFunc({ ...params, latestBuild: false }), | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @param {{ | ||||||
|  |  *   compiler: import("webpack").Compiler, | ||||||
|  |  *   contentBase: string, | ||||||
|  |  *   port: number, | ||||||
|  |  *   listenHost?: string | ||||||
|  |  * }} | ||||||
|  |  */ | ||||||
| const runDevServer = ({ | const runDevServer = ({ | ||||||
|   compiler, |   compiler, | ||||||
|   contentBase, |   contentBase, | ||||||
| @@ -33,24 +41,27 @@ const runDevServer = ({ | |||||||
|       throw err; |       throw err; | ||||||
|     } |     } | ||||||
|     // Server listening |     // Server listening | ||||||
|     log("[webpack-dev-server]", `http://localhost:${port}`); |     log( | ||||||
|  |       "[webpack-dev-server]", | ||||||
|  |       `Project is running at http://localhost:${port}` | ||||||
|  |     ); | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
| const handler = (done) => (err, stats) => { | const handler = (done) => (err, stats) => { | ||||||
|   if (err) { |   if (err) { | ||||||
|     console.log(err.stack || err); |     log.error(err.stack || err); | ||||||
|     if (err.details) { |     if (err.details) { | ||||||
|       console.log(err.details); |       log.error(err.details); | ||||||
|     } |     } | ||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   log(`Build done @ ${new Date().toLocaleTimeString()}`); |  | ||||||
|  |  | ||||||
|   if (stats.hasErrors() || stats.hasWarnings()) { |   if (stats.hasErrors() || stats.hasWarnings()) { | ||||||
|     console.log(stats.toString("minimal")); |     console.log(stats.toString("minimal")); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   log(`Build done @ ${new Date().toLocaleTimeString()}`); | ||||||
|  |  | ||||||
|   if (done) { |   if (done) { | ||||||
|     done(); |     done(); | ||||||
|   } |   } | ||||||
| @@ -64,7 +75,7 @@ gulp.task("webpack-watch-app", () => { | |||||||
|   ); |   ); | ||||||
|   gulp.watch( |   gulp.watch( | ||||||
|     path.join(paths.translations_src, "en.json"), |     path.join(paths.translations_src, "en.json"), | ||||||
|     gulp.series("build-translations", "copy-translations") |     gulp.series("build-translations", "copy-translations-app") | ||||||
|   ); |   ); | ||||||
| }); | }); | ||||||
|  |  | ||||||
| @@ -82,7 +93,7 @@ gulp.task( | |||||||
| gulp.task("webpack-dev-server-demo", () => { | gulp.task("webpack-dev-server-demo", () => { | ||||||
|   runDevServer({ |   runDevServer({ | ||||||
|     compiler: webpack(bothBuilds(createDemoConfig, { isProdBuild: false })), |     compiler: webpack(bothBuilds(createDemoConfig, { isProdBuild: false })), | ||||||
|     contentBase: paths.demo_root, |     contentBase: paths.demo_output_root, | ||||||
|     port: 8090, |     port: 8090, | ||||||
|   }); |   }); | ||||||
| }); | }); | ||||||
| @@ -103,7 +114,7 @@ gulp.task( | |||||||
| gulp.task("webpack-dev-server-cast", () => { | gulp.task("webpack-dev-server-cast", () => { | ||||||
|   runDevServer({ |   runDevServer({ | ||||||
|     compiler: webpack(bothBuilds(createCastConfig, { isProdBuild: false })), |     compiler: webpack(bothBuilds(createCastConfig, { isProdBuild: false })), | ||||||
|     contentBase: paths.cast_root, |     contentBase: paths.cast_output_root, | ||||||
|     port: 8080, |     port: 8080, | ||||||
|     // Accessible from the network, because that's how Cast hits it. |     // Accessible from the network, because that's how Cast hits it. | ||||||
|     listenHost: "0.0.0.0", |     listenHost: "0.0.0.0", | ||||||
| @@ -129,7 +140,7 @@ gulp.task("webpack-watch-hassio", () => { | |||||||
|   webpack( |   webpack( | ||||||
|     createHassioConfig({ |     createHassioConfig({ | ||||||
|       isProdBuild: false, |       isProdBuild: false, | ||||||
|       latestBuild: false, |       latestBuild: true, | ||||||
|     }) |     }) | ||||||
|   ).watch({}, handler()); |   ).watch({}, handler()); | ||||||
| }); | }); | ||||||
| @@ -139,9 +150,8 @@ gulp.task( | |||||||
|   () => |   () => | ||||||
|     new Promise((resolve) => |     new Promise((resolve) => | ||||||
|       webpack( |       webpack( | ||||||
|         createHassioConfig({ |         bothBuilds(createHassioConfig, { | ||||||
|           isProdBuild: true, |           isProdBuild: true, | ||||||
|           latestBuild: false, |  | ||||||
|         }), |         }), | ||||||
|         handler(resolve) |         handler(resolve) | ||||||
|       ) |       ) | ||||||
| @@ -152,7 +162,7 @@ gulp.task("webpack-dev-server-gallery", () => { | |||||||
|   runDevServer({ |   runDevServer({ | ||||||
|     // We don't use the es5 build, but the dev server will fuck up the publicPath if we don't |     // We don't use the es5 build, but the dev server will fuck up the publicPath if we don't | ||||||
|     compiler: webpack(bothBuilds(createGalleryConfig, { isProdBuild: false })), |     compiler: webpack(bothBuilds(createGalleryConfig, { isProdBuild: false })), | ||||||
|     contentBase: paths.gallery_root, |     contentBase: paths.gallery_output_root, | ||||||
|     port: 8100, |     port: 8100, | ||||||
|   }); |   }); | ||||||
| }); | }); | ||||||
|   | |||||||
| @@ -1,34 +1,45 @@ | |||||||
| var path = require("path"); | const path = require("path"); | ||||||
|  |  | ||||||
| module.exports = { | module.exports = { | ||||||
|   polymer_dir: path.resolve(__dirname, ".."), |   polymer_dir: path.resolve(__dirname, ".."), | ||||||
|  |  | ||||||
|   build_dir: path.resolve(__dirname, "../build"), |   build_dir: path.resolve(__dirname, "../build"), | ||||||
|   root: path.resolve(__dirname, "../hass_frontend"), |   app_output_root: path.resolve(__dirname, "../hass_frontend"), | ||||||
|   static: path.resolve(__dirname, "../hass_frontend/static"), |   app_output_static: path.resolve(__dirname, "../hass_frontend/static"), | ||||||
|   output: path.resolve(__dirname, "../hass_frontend/frontend_latest"), |   app_output_latest: path.resolve( | ||||||
|   output_es5: path.resolve(__dirname, "../hass_frontend/frontend_es5"), |     __dirname, | ||||||
|  |     "../hass_frontend/frontend_latest" | ||||||
|  |   ), | ||||||
|  |   app_output_es5: path.resolve(__dirname, "../hass_frontend/frontend_es5"), | ||||||
|  |  | ||||||
|   demo_dir: path.resolve(__dirname, "../demo"), |   demo_dir: path.resolve(__dirname, "../demo"), | ||||||
|   demo_root: path.resolve(__dirname, "../demo/dist"), |   demo_output_root: path.resolve(__dirname, "../demo/dist"), | ||||||
|   demo_static: path.resolve(__dirname, "../demo/dist/static"), |   demo_output_static: path.resolve(__dirname, "../demo/dist/static"), | ||||||
|   demo_output: path.resolve(__dirname, "../demo/dist/frontend_latest"), |   demo_output_latest: path.resolve(__dirname, "../demo/dist/frontend_latest"), | ||||||
|   demo_output_es5: path.resolve(__dirname, "../demo/dist/frontend_es5"), |   demo_output_es5: path.resolve(__dirname, "../demo/dist/frontend_es5"), | ||||||
|  |  | ||||||
|   cast_dir: path.resolve(__dirname, "../cast"), |   cast_dir: path.resolve(__dirname, "../cast"), | ||||||
|   cast_root: path.resolve(__dirname, "../cast/dist"), |   cast_output_root: path.resolve(__dirname, "../cast/dist"), | ||||||
|   cast_static: path.resolve(__dirname, "../cast/dist/static"), |   cast_output_static: path.resolve(__dirname, "../cast/dist/static"), | ||||||
|   cast_output: path.resolve(__dirname, "../cast/dist/frontend_latest"), |   cast_output_latest: path.resolve(__dirname, "../cast/dist/frontend_latest"), | ||||||
|   cast_output_es5: path.resolve(__dirname, "../cast/dist/frontend_es5"), |   cast_output_es5: path.resolve(__dirname, "../cast/dist/frontend_es5"), | ||||||
|  |  | ||||||
|   gallery_dir: path.resolve(__dirname, "../gallery"), |   gallery_dir: path.resolve(__dirname, "../gallery"), | ||||||
|   gallery_root: path.resolve(__dirname, "../gallery/dist"), |   gallery_output_root: path.resolve(__dirname, "../gallery/dist"), | ||||||
|   gallery_output: path.resolve(__dirname, "../gallery/dist/frontend_latest"), |   gallery_output_latest: path.resolve( | ||||||
|   gallery_static: path.resolve(__dirname, "../gallery/dist/static"), |     __dirname, | ||||||
|  |     "../gallery/dist/frontend_latest" | ||||||
|  |   ), | ||||||
|  |   gallery_output_static: path.resolve(__dirname, "../gallery/dist/static"), | ||||||
|  |  | ||||||
|   hassio_dir: path.resolve(__dirname, "../hassio"), |   hassio_dir: path.resolve(__dirname, "../hassio"), | ||||||
|   hassio_root: path.resolve(__dirname, "../hassio/build"), |   hassio_output_root: path.resolve(__dirname, "../hassio/build"), | ||||||
|   hassio_publicPath: "/api/hassio/app/", |   hassio_output_latest: path.resolve( | ||||||
|  |     __dirname, | ||||||
|  |     "../hassio/build/frontend_latest" | ||||||
|  |   ), | ||||||
|  |   hassio_output_es5: path.resolve(__dirname, "../hassio/build/frontend_es5"), | ||||||
|  |   hassio_publicPath: "/api/hassio/app", | ||||||
|  |  | ||||||
|   translations_src: path.resolve(__dirname, "../src/translations"), |   translations_src: path.resolve(__dirname, "../src/translations"), | ||||||
| }; | }; | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								build-scripts/removedIcons.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | |||||||
|  | [] | ||||||
							
								
								
									
										14
									
								
								build-scripts/rollup-plugins/dont-hash-plugin.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | |||||||
|  | module.exports = function (opts = {}) { | ||||||
|  |   const dontHash = opts.dontHash || new Set(); | ||||||
|  |  | ||||||
|  |   return { | ||||||
|  |     name: "dont-hash", | ||||||
|  |     renderChunk(_code, chunk, _options) { | ||||||
|  |       if (!chunk.isEntry || !dontHash.has(chunk.name)) { | ||||||
|  |         return null; | ||||||
|  |       } | ||||||
|  |       chunk.fileName = `${chunk.name}.js`; | ||||||
|  |       return null; | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
							
								
								
									
										26
									
								
								build-scripts/rollup-plugins/ignore-plugin.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | |||||||
|  | const path = require("path"); | ||||||
|  |  | ||||||
|  | module.exports = function (userOptions = {}) { | ||||||
|  |   // Files need to be absolute paths. | ||||||
|  |   // This only works if the file has no exports | ||||||
|  |   // and only is imported for its side effects | ||||||
|  |   const files = userOptions.files || []; | ||||||
|  |  | ||||||
|  |   if (files.length === 0) { | ||||||
|  |     return { | ||||||
|  |       name: "ignore", | ||||||
|  |     }; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return { | ||||||
|  |     name: "ignore", | ||||||
|  |  | ||||||
|  |     load(id) { | ||||||
|  |       return files.some((toIgnorePath) => id.startsWith(toIgnorePath)) | ||||||
|  |         ? { | ||||||
|  |             code: "", | ||||||
|  |           } | ||||||
|  |         : null; | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
							
								
								
									
										34
									
								
								build-scripts/rollup-plugins/manifest-plugin.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | |||||||
|  | const url = require("url"); | ||||||
|  |  | ||||||
|  | const defaultOptions = { | ||||||
|  |   publicPath: "", | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | module.exports = function (userOptions = {}) { | ||||||
|  |   const options = { ...defaultOptions, ...userOptions }; | ||||||
|  |  | ||||||
|  |   return { | ||||||
|  |     name: "manifest", | ||||||
|  |     generateBundle(outputOptions, bundle) { | ||||||
|  |       const manifest = {}; | ||||||
|  |  | ||||||
|  |       for (const chunk of Object.values(bundle)) { | ||||||
|  |         if (!chunk.isEntry) { | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |         // Add js extension to mimic Webpack manifest. | ||||||
|  |         manifest[`${chunk.name}.js`] = url.resolve( | ||||||
|  |           options.publicPath, | ||||||
|  |           chunk.fileName | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       this.emitFile({ | ||||||
|  |         type: "asset", | ||||||
|  |         source: JSON.stringify(manifest, undefined, 2), | ||||||
|  |         name: "manifest.json", | ||||||
|  |         fileName: "manifest.json", | ||||||
|  |       }); | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
							
								
								
									
										149
									
								
								build-scripts/rollup-plugins/worker-plugin.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,149 @@ | |||||||
|  | // Worker plugin | ||||||
|  | // Each worker will include all of its dependencies | ||||||
|  | // instead of relying on an importer. | ||||||
|  |  | ||||||
|  | // Forked from v.1.4.1 | ||||||
|  | // https://github.com/surma/rollup-plugin-off-main-thread | ||||||
|  | /** | ||||||
|  |  * Copyright 2018 Google Inc. All Rights Reserved. | ||||||
|  |  * Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  |  * you may not use this file except in compliance with the License. | ||||||
|  |  * You may obtain a copy of the License at | ||||||
|  |  *     http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  |  * Unless required by applicable law or agreed to in writing, software | ||||||
|  |  * distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  |  * See the License for the specific language governing permissions and | ||||||
|  |  * limitations under the License. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | const rollup = require("rollup"); | ||||||
|  | const path = require("path"); | ||||||
|  | const MagicString = require("magic-string"); | ||||||
|  |  | ||||||
|  | const defaultOpts = { | ||||||
|  |   // A RegExp to find `new Workers()` calls. The second capture group _must_ | ||||||
|  |   // capture the provided file name without the quotes. | ||||||
|  |   workerRegexp: /new Worker\((["'])(.+?)\1(,[^)]+)?\)/g, | ||||||
|  |   plugins: ["node-resolve", "commonjs", "babel", "terser", "ignore"], | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | async function getBundledWorker(workerPath, rollupOptions) { | ||||||
|  |   const bundle = await rollup.rollup({ | ||||||
|  |     ...rollupOptions, | ||||||
|  |     input: { | ||||||
|  |       worker: workerPath, | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  |   const { output } = await bundle.generate({ | ||||||
|  |     // Generates cleanest output, we shouldn't have any imports/exports | ||||||
|  |     // that would be incompatible with ES5. | ||||||
|  |     format: "es", | ||||||
|  |     // We should not export anything. This will fail build if we are. | ||||||
|  |     exports: "none", | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   let code; | ||||||
|  |  | ||||||
|  |   for (const chunkOrAsset of output) { | ||||||
|  |     if (chunkOrAsset.name === "worker") { | ||||||
|  |       code = chunkOrAsset.code; | ||||||
|  |     } else if (chunkOrAsset.type !== "asset") { | ||||||
|  |       throw new Error("Unexpected extra output"); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return code; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | module.exports = function (opts = {}) { | ||||||
|  |   opts = { ...defaultOpts, ...opts }; | ||||||
|  |  | ||||||
|  |   let rollupOptions; | ||||||
|  |   let refIds; | ||||||
|  |  | ||||||
|  |   return { | ||||||
|  |     name: "hass-worker", | ||||||
|  |  | ||||||
|  |     async buildStart(options) { | ||||||
|  |       refIds = {}; | ||||||
|  |       rollupOptions = { | ||||||
|  |         plugins: options.plugins.filter((plugin) => | ||||||
|  |           opts.plugins.includes(plugin.name) | ||||||
|  |         ), | ||||||
|  |       }; | ||||||
|  |     }, | ||||||
|  |  | ||||||
|  |     async transform(code, id) { | ||||||
|  |       // Copy the regexp as they are stateful and this hook is async. | ||||||
|  |       const workerRegexp = new RegExp( | ||||||
|  |         opts.workerRegexp.source, | ||||||
|  |         opts.workerRegexp.flags | ||||||
|  |       ); | ||||||
|  |       if (!workerRegexp.test(code)) { | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const ms = new MagicString(code); | ||||||
|  |       // Reset the regexp | ||||||
|  |       workerRegexp.lastIndex = 0; | ||||||
|  |       while (true) { | ||||||
|  |         const match = workerRegexp.exec(code); | ||||||
|  |         if (!match) { | ||||||
|  |           break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const workerFile = match[2]; | ||||||
|  |         let optionsObject = {}; | ||||||
|  |         // Parse the optional options object | ||||||
|  |         if (match[3] && match[3].length > 0) { | ||||||
|  |           // FIXME: ooooof! | ||||||
|  |           optionsObject = new Function(`return ${match[3].slice(1)};`)(); | ||||||
|  |         } | ||||||
|  |         delete optionsObject.type; | ||||||
|  |  | ||||||
|  |         if (!new RegExp("^.*/").test(workerFile)) { | ||||||
|  |           this.warn( | ||||||
|  |             `Paths passed to the Worker constructor must be relative or absolute, i.e. start with /, ./ or ../ (just like dynamic import!). Ignoring "${workerFile}".` | ||||||
|  |           ); | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Find worker file and store it as a chunk with ID prefixed for our loader | ||||||
|  |         const resolvedWorkerFile = (await this.resolve(workerFile, id)).id; | ||||||
|  |         let chunkRefId; | ||||||
|  |         if (resolvedWorkerFile in refIds) { | ||||||
|  |           chunkRefId = refIds[resolvedWorkerFile]; | ||||||
|  |         } else { | ||||||
|  |           this.addWatchFile(resolvedWorkerFile); | ||||||
|  |           const source = await getBundledWorker( | ||||||
|  |             resolvedWorkerFile, | ||||||
|  |             rollupOptions | ||||||
|  |           ); | ||||||
|  |           chunkRefId = refIds[resolvedWorkerFile] = this.emitFile({ | ||||||
|  |             name: path.basename(resolvedWorkerFile), | ||||||
|  |             source, | ||||||
|  |             type: "asset", | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const workerParametersStartIndex = match.index + "new Worker(".length; | ||||||
|  |         const workerParametersEndIndex = | ||||||
|  |           match.index + match[0].length - ")".length; | ||||||
|  |  | ||||||
|  |         ms.overwrite( | ||||||
|  |           workerParametersStartIndex, | ||||||
|  |           workerParametersEndIndex, | ||||||
|  |           `import.meta.ROLLUP_FILE_URL_${chunkRefId}, ${JSON.stringify( | ||||||
|  |             optionsObject | ||||||
|  |           )}` | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       return { | ||||||
|  |         code: ms.toString(), | ||||||
|  |         map: ms.generateMap({ hires: true }), | ||||||
|  |       }; | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
							
								
								
									
										151
									
								
								build-scripts/rollup.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,151 @@ | |||||||
|  | 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 replace = require("@rollup/plugin-replace"); | ||||||
|  | const visualizer = require("rollup-plugin-visualizer"); | ||||||
|  | const { string } = require("rollup-plugin-string"); | ||||||
|  | const { terser } = require("rollup-plugin-terser"); | ||||||
|  | const manifest = require("./rollup-plugins/manifest-plugin"); | ||||||
|  | const worker = require("./rollup-plugins/worker-plugin"); | ||||||
|  | const dontHashPlugin = require("./rollup-plugins/dont-hash-plugin"); | ||||||
|  | const ignore = require("./rollup-plugins/ignore-plugin"); | ||||||
|  |  | ||||||
|  | const bundle = require("./bundle"); | ||||||
|  | const paths = require("./paths"); | ||||||
|  |  | ||||||
|  | const extensions = [".js", ".ts"]; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @param {Object} arg | ||||||
|  |  * @param { import("rollup").InputOption } arg.input | ||||||
|  |  */ | ||||||
|  | const createRollupConfig = ({ | ||||||
|  |   entry, | ||||||
|  |   outputPath, | ||||||
|  |   defineOverlay, | ||||||
|  |   isProdBuild, | ||||||
|  |   latestBuild, | ||||||
|  |   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 }) | ||||||
|  |         ), | ||||||
|  |         manifest({ | ||||||
|  |           publicPath, | ||||||
|  |         }), | ||||||
|  |         worker(), | ||||||
|  |         dontHashPlugin({ dontHash }), | ||||||
|  |         isProdBuild && terser(bundle.terserOptions(latestBuild)), | ||||||
|  |         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", | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | ||||||
|  |   return createRollupConfig( | ||||||
|  |     bundle.config.app({ | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       isStatsBuild, | ||||||
|  |     }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | ||||||
|  |   return createRollupConfig( | ||||||
|  |     bundle.config.demo({ | ||||||
|  |       isProdBuild, | ||||||
|  |       latestBuild, | ||||||
|  |       isStatsBuild, | ||||||
|  |     }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createCastConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createRollupConfig(bundle.config.cast({ isProdBuild, latestBuild })); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createHassioConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createRollupConfig(bundle.config.hassio({ isProdBuild, latestBuild })); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const createGalleryConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|  |   return createRollupConfig( | ||||||
|  |     bundle.config.gallery({ isProdBuild, latestBuild }) | ||||||
|  |   ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | module.exports = { | ||||||
|  |   createAppConfig, | ||||||
|  |   createDemoConfig, | ||||||
|  |   createCastConfig, | ||||||
|  |   createHassioConfig, | ||||||
|  |   createGalleryConfig, | ||||||
|  | }; | ||||||
| @@ -2,14 +2,28 @@ const webpack = require("webpack"); | |||||||
| const path = require("path"); | const path = require("path"); | ||||||
| const TerserPlugin = require("terser-webpack-plugin"); | const TerserPlugin = require("terser-webpack-plugin"); | ||||||
| const ManifestPlugin = require("webpack-manifest-plugin"); | const ManifestPlugin = require("webpack-manifest-plugin"); | ||||||
| const WorkerPlugin = require("worker-plugin"); |  | ||||||
| const paths = require("./paths.js"); | const paths = require("./paths.js"); | ||||||
| const env = require("./env.js"); | const bundle = require("./bundle"); | ||||||
| const { babelLoaderConfig } = require("./babel.js"); | const log = require("fancy-log"); | ||||||
|  |  | ||||||
|  | class LogStartCompilePlugin { | ||||||
|  |   ignoredFirst = false; | ||||||
|  |  | ||||||
|  |   apply(compiler) { | ||||||
|  |     compiler.hooks.beforeCompile.tap("LogStartCompilePlugin", () => { | ||||||
|  |       if (!this.ignoredFirst) { | ||||||
|  |         this.ignoredFirst = true; | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       log("Changes detected. Starting compilation"); | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
| const createWebpackConfig = ({ | const createWebpackConfig = ({ | ||||||
|   entry, |   entry, | ||||||
|   outputRoot, |   outputPath, | ||||||
|  |   publicPath, | ||||||
|   defineOverlay, |   defineOverlay, | ||||||
|   isProdBuild, |   isProdBuild, | ||||||
|   latestBuild, |   latestBuild, | ||||||
| @@ -19,74 +33,93 @@ const createWebpackConfig = ({ | |||||||
|   if (!dontHash) { |   if (!dontHash) { | ||||||
|     dontHash = new Set(); |     dontHash = new Set(); | ||||||
|   } |   } | ||||||
|  |   const ignorePackages = bundle.ignorePackages({ latestBuild }); | ||||||
|   return { |   return { | ||||||
|     mode: isProdBuild ? "production" : "development", |     mode: isProdBuild ? "production" : "development", | ||||||
|     devtool: isProdBuild |     devtool: isProdBuild | ||||||
|       ? "cheap-module-source-map" |       ? "cheap-module-source-map" | ||||||
|       : "eval-cheap-module-source-map", |       : "eval-cheap-module-source-map", | ||||||
|     entry, |     entry, | ||||||
|  |     node: false, | ||||||
|     module: { |     module: { | ||||||
|       rules: [ |       rules: [ | ||||||
|         babelLoaderConfig({ latestBuild }), |         { | ||||||
|  |           test: /\.m?js$|\.ts$/, | ||||||
|  |           exclude: bundle.babelExclude(), | ||||||
|  |           use: { | ||||||
|  |             loader: "babel-loader", | ||||||
|  |             options: bundle.babelOptions({ latestBuild }), | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|         { |         { | ||||||
|           test: /\.css$/, |           test: /\.css$/, | ||||||
|           use: "raw-loader", |           use: "raw-loader", | ||||||
|         }, |         }, | ||||||
|       ], |       ], | ||||||
|     }, |     }, | ||||||
|     externals: { |  | ||||||
|       esprima: "esprima", |  | ||||||
|     }, |  | ||||||
|     optimization: { |     optimization: { | ||||||
|       minimizer: [ |       minimizer: [ | ||||||
|         new TerserPlugin({ |         new TerserPlugin({ | ||||||
|           cache: true, |  | ||||||
|           parallel: true, |           parallel: true, | ||||||
|           extractComments: true, |           extractComments: true, | ||||||
|           sourceMap: true, |           terserOptions: bundle.terserOptions(latestBuild), | ||||||
|           terserOptions: { |  | ||||||
|             safari10: true, |  | ||||||
|             ecma: latestBuild ? undefined : 5, |  | ||||||
|           }, |  | ||||||
|         }), |         }), | ||||||
|       ], |       ], | ||||||
|     }, |     }, | ||||||
|     plugins: [ |     plugins: [ | ||||||
|       new WorkerPlugin(), |       new ManifestPlugin({ | ||||||
|       new ManifestPlugin(), |         // Only include the JS of entrypoints | ||||||
|       new webpack.DefinePlugin({ |         filter: (file) => file.isInitial && !file.name.endsWith(".map"), | ||||||
|         __DEV__: !isProdBuild, |       }), | ||||||
|         __BUILD__: JSON.stringify(latestBuild ? "latest" : "es5"), |       new webpack.DefinePlugin( | ||||||
|         __VERSION__: JSON.stringify(env.version()), |         bundle.definedVars({ isProdBuild, latestBuild, defineOverlay }) | ||||||
|         __DEMO__: false, |       ), | ||||||
|         __BACKWARDS_COMPAT__: false, |       new webpack.IgnorePlugin({ | ||||||
|         __STATIC_PATH__: "/static/", |         checkResource(resource, context) { | ||||||
|         "process.env.NODE_ENV": JSON.stringify( |           // Only use ignore to intercept imports that we don't control | ||||||
|           isProdBuild ? "production" : "development" |           // inside node_module dependencies. | ||||||
|         ), |           if ( | ||||||
|         ...defineOverlay, |             !context.includes("/node_modules/") || | ||||||
|  |             // calling define.amd will call require("!!webpack amd options") | ||||||
|  |             resource.startsWith("!!webpack") || | ||||||
|  |             // loaded by webpack dev server but doesn't exist. | ||||||
|  |             resource === "webpack/hot" | ||||||
|  |           ) { | ||||||
|  |             return false; | ||||||
|  |           } | ||||||
|  |           let fullPath; | ||||||
|  |           try { | ||||||
|  |             fullPath = resource.startsWith(".") | ||||||
|  |               ? path.resolve(context, resource) | ||||||
|  |               : require.resolve(resource); | ||||||
|  |           } catch (err) { | ||||||
|  |             console.error( | ||||||
|  |               "Error in Home Assistant ignore plugin", | ||||||
|  |               resource, | ||||||
|  |               context | ||||||
|  |             ); | ||||||
|  |             throw err; | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |           return ignorePackages.some((toIgnorePath) => | ||||||
|  |             fullPath.startsWith(toIgnorePath) | ||||||
|  |           ); | ||||||
|  |         }, | ||||||
|       }), |       }), | ||||||
|       // Ignore moment.js locales |  | ||||||
|       new webpack.IgnorePlugin(/^\.\/locale$/, /moment$/), |  | ||||||
|       // Color.js is bloated, it contains all color definitions for all material color sets. |  | ||||||
|       new webpack.NormalModuleReplacementPlugin( |       new webpack.NormalModuleReplacementPlugin( | ||||||
|         /@polymer\/paper-styles\/color\.js$/, |         new RegExp(bundle.emptyPackages({ latestBuild }).join("|")), | ||||||
|         path.resolve(paths.polymer_dir, "src/util/empty.js") |         path.resolve(paths.polymer_dir, "src/util/empty.js") | ||||||
|       ), |       ), | ||||||
|       // Ignore roboto pointing at CDN. We use local font-roboto-local. |       // 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 webpack.NormalModuleReplacementPlugin( | ||||||
|         /@polymer\/font-roboto\/roboto\.js$/, |         new RegExp( | ||||||
|         path.resolve(paths.polymer_dir, "src/util/empty.js") |           require.resolve( | ||||||
|  |             "lit-virtualizer/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js" | ||||||
|  |           ) | ||||||
|         ), |         ), | ||||||
|       new webpack.NormalModuleReplacementPlugin( |         path.resolve(paths.polymer_dir, "src/resources/EventTarget-ponyfill.js") | ||||||
|         /@vaadin\/vaadin-material-styles\/font-roboto\.js$/, |  | ||||||
|         path.resolve(paths.polymer_dir, "src/util/empty.js") |  | ||||||
|       ), |  | ||||||
|       // Ignore mwc icons pointing at CDN. |  | ||||||
|       new webpack.NormalModuleReplacementPlugin( |  | ||||||
|         /@material\/mwc-icon\/mwc-icon-font\.js$/, |  | ||||||
|         path.resolve(paths.polymer_dir, "src/util/empty.js") |  | ||||||
|       ), |       ), | ||||||
|  |       !isProdBuild && new LogStartCompilePlugin(), | ||||||
|     ].filter(Boolean), |     ].filter(Boolean), | ||||||
|     resolve: { |     resolve: { | ||||||
|       extensions: [".ts", ".js", ".json"], |       extensions: [".ts", ".js", ".json"], | ||||||
| @@ -98,15 +131,28 @@ const createWebpackConfig = ({ | |||||||
|         } |         } | ||||||
|         return `${chunk.name}.${chunk.hash.substr(0, 8)}.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: |       chunkFilename: | ||||||
|         isProdBuild && !isStatsBuild |         isProdBuild && !isStatsBuild | ||||||
|           ? "chunk.[chunkhash].js" |           ? "chunk.[chunkhash].js" | ||||||
|           : "[name].chunk.js", |           : "[name].chunk.js", | ||||||
|       path: path.resolve( |       path: outputPath, | ||||||
|         outputRoot, |       publicPath, | ||||||
|         latestBuild ? "frontend_latest" : "frontend_es5" |  | ||||||
|       ), |  | ||||||
|       publicPath: latestBuild ? "/frontend_latest/" : "/frontend_es5/", |  | ||||||
|       // To silence warning in worker plugin |       // To silence warning in worker plugin | ||||||
|       globalObject: "self", |       globalObject: "self", | ||||||
|     }, |     }, | ||||||
| @@ -114,94 +160,31 @@ const createWebpackConfig = ({ | |||||||
| }; | }; | ||||||
|  |  | ||||||
| const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | ||||||
|   return createWebpackConfig({ |   return createWebpackConfig( | ||||||
|     entry: { |     bundle.config.app({ isProdBuild, latestBuild, isStatsBuild }) | ||||||
|       service_worker: "./src/entrypoints/service_worker.ts", |   ); | ||||||
|       app: "./src/entrypoints/app.ts", |  | ||||||
|       authorize: "./src/entrypoints/authorize.ts", |  | ||||||
|       onboarding: "./src/entrypoints/onboarding.ts", |  | ||||||
|       core: "./src/entrypoints/core.ts", |  | ||||||
|       compatibility: "./src/entrypoints/compatibility.ts", |  | ||||||
|       "custom-panel": "./src/entrypoints/custom-panel.ts", |  | ||||||
|     }, |  | ||||||
|     outputRoot: paths.root, |  | ||||||
|     isProdBuild, |  | ||||||
|     latestBuild, |  | ||||||
|     isStatsBuild, |  | ||||||
|   }); |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => { | ||||||
|   return createWebpackConfig({ |   return createWebpackConfig( | ||||||
|     entry: { |     bundle.config.demo({ isProdBuild, latestBuild, isStatsBuild }) | ||||||
|       main: path.resolve(paths.demo_dir, "src/entrypoint.ts"), |   ); | ||||||
|       compatibility: path.resolve( |  | ||||||
|         paths.polymer_dir, |  | ||||||
|         "src/entrypoints/compatibility.ts" |  | ||||||
|       ), |  | ||||||
|     }, |  | ||||||
|     outputRoot: paths.demo_root, |  | ||||||
|     defineOverlay: { |  | ||||||
|       __VERSION__: JSON.stringify(`DEMO-${env.version()}`), |  | ||||||
|       __DEMO__: true, |  | ||||||
|     }, |  | ||||||
|     isProdBuild, |  | ||||||
|     latestBuild, |  | ||||||
|     isStatsBuild, |  | ||||||
|   }); |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const createCastConfig = ({ isProdBuild, latestBuild }) => { | const createCastConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|   const entry = { |   return createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild })); | ||||||
|     launcher: path.resolve(paths.cast_dir, "src/launcher/entrypoint.ts"), |  | ||||||
|   }; |  | ||||||
|  |  | ||||||
|   if (latestBuild) { |  | ||||||
|     entry.receiver = path.resolve(paths.cast_dir, "src/receiver/entrypoint.ts"); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return createWebpackConfig({ |  | ||||||
|     entry, |  | ||||||
|     outputRoot: paths.cast_root, |  | ||||||
|     isProdBuild, |  | ||||||
|     latestBuild, |  | ||||||
|     defineOverlay: { |  | ||||||
|       __BACKWARDS_COMPAT__: true, |  | ||||||
|     }, |  | ||||||
|   }); |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const createHassioConfig = ({ isProdBuild, latestBuild }) => { | const createHassioConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|   if (latestBuild) { |   return createWebpackConfig( | ||||||
|     throw new Error("Hass.io does not support latest build!"); |     bundle.config.hassio({ isProdBuild, latestBuild }) | ||||||
|   } |   ); | ||||||
|   const config = createWebpackConfig({ |  | ||||||
|     entry: { |  | ||||||
|       entrypoint: path.resolve(paths.hassio_dir, "src/entrypoint.ts"), |  | ||||||
|     }, |  | ||||||
|     outputRoot: "", |  | ||||||
|     isProdBuild, |  | ||||||
|     latestBuild, |  | ||||||
|     dontHash: new Set(["entrypoint"]), |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   config.output.path = paths.hassio_root; |  | ||||||
|   config.output.publicPath = paths.hassio_publicPath; |  | ||||||
|  |  | ||||||
|   return config; |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| const createGalleryConfig = ({ isProdBuild, latestBuild }) => { | const createGalleryConfig = ({ isProdBuild, latestBuild }) => { | ||||||
|   const config = createWebpackConfig({ |   return createWebpackConfig( | ||||||
|     entry: { |     bundle.config.gallery({ isProdBuild, latestBuild }) | ||||||
|       entrypoint: path.resolve(paths.gallery_dir, "src/entrypoint.js"), |   ); | ||||||
|     }, |  | ||||||
|     outputRoot: paths.gallery_root, |  | ||||||
|     isProdBuild, |  | ||||||
|     latestBuild, |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   return config; |  | ||||||
| }; | }; | ||||||
|  |  | ||||||
| module.exports = { | module.exports = { | ||||||
|   | |||||||
| Before Width: | Height: | Size: 18 KiB After Width: | Height: | Size: 16 KiB | 
| Before Width: | Height: | Size: 186 KiB After Width: | Height: | Size: 186 KiB | 
| Before Width: | Height: | Size: 68 KiB After Width: | Height: | Size: 68 KiB | 
							
								
								
									
										10
									
								
								cast/rollup.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | |||||||
|  | const rollup = require("../build-scripts/rollup.js"); | ||||||
|  | const env = require("../build-scripts/env.js"); | ||||||
|  |  | ||||||
|  | const config = rollup.createCastConfig({ | ||||||
|  |   isProdBuild: env.isProdBuild(), | ||||||
|  |   latestBuild: true, | ||||||
|  |   isStatsBuild: env.isStatsBuild(), | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | module.exports = { ...config.inputOptions, output: config.outputOptions }; | ||||||
| @@ -37,18 +37,21 @@ | |||||||
|   <body> |   <body> | ||||||
|     <%= renderTemplate('_js_base') %> |     <%= renderTemplate('_js_base') %> | ||||||
|  |  | ||||||
|     <script type="module" crossorigin="use-credentials"> |     <script> | ||||||
|       import "<%= latestLauncherJS %>"; |       import("<%= latestLauncherJS %>"); | ||||||
|  |       window.latestJS = true; | ||||||
|     </script> |     </script> | ||||||
|  |  | ||||||
|     <script nomodule> |     <script> | ||||||
|       (function() { |       if (!window.latestJS) { | ||||||
|         // // Safari 10.1 supports type=module but ignores nomodule, so we add this check. |         <% if (useRollup) { %> | ||||||
|         if (!isS101) { |           _ls("/static/js/s.min.js").onload = function() { | ||||||
|           _ls("/static/polyfills/custom-elements-es5-adapter.js"); |             System.import("<%= es5LauncherJS %>"); | ||||||
|  |           }; | ||||||
|  |         <% } else { %> | ||||||
|           _ls("<%= es5LauncherJS %>"); |           _ls("<%= es5LauncherJS %>"); | ||||||
|  |         <% } %> | ||||||
|       } |       } | ||||||
|       })(); |  | ||||||
|     </script> |     </script> | ||||||
|  |  | ||||||
|     <hc-layout subtitle="FAQ"> |     <hc-layout subtitle="FAQ"> | ||||||
| @@ -209,13 +212,8 @@ | |||||||
|           Chromecast is a technology developed by Google, and is available on: |           Chromecast is a technology developed by Google, and is available on: | ||||||
|         </p> |         </p> | ||||||
|         <ul> |         <ul> | ||||||
|           <li>Google Chrome (all platforms except on iOS)</li> |           <li>Google Chrome (all platforms except iOS)</li> | ||||||
|           <li> |           <li>Microsoft Edge (all platforms)</li> | ||||||
|             Microsoft Edge (all platforms, |  | ||||||
|             <a href="https://www.microsoftedgeinsider.com" target="_blank" |  | ||||||
|               >dev and canary builds only</a |  | ||||||
|             >) |  | ||||||
|           </li> |  | ||||||
|         </ul> |         </ul> | ||||||
|       </div> |       </div> | ||||||
|  |  | ||||||
| @@ -248,7 +246,7 @@ http: | |||||||
|  |  | ||||||
|     <script> |     <script> | ||||||
|       var _gaq = [["_setAccount", "UA-57927901-9"], ["_trackPageview"]]; |       var _gaq = [["_setAccount", "UA-57927901-9"], ["_trackPageview"]]; | ||||||
|       (function(d, t) { |       (function (d, t) { | ||||||
|         var g = d.createElement(t), |         var g = d.createElement(t), | ||||||
|           s = d.getElementsByTagName(t)[0]; |           s = d.getElementsByTagName(t)[0]; | ||||||
|         g.src = |         g.src = | ||||||
|   | |||||||
| @@ -28,18 +28,21 @@ | |||||||
|  |  | ||||||
|     <hc-connect></hc-connect> |     <hc-connect></hc-connect> | ||||||
|  |  | ||||||
|     <script type="module" crossorigin="use-credentials"> |     <script> | ||||||
|       import "<%= latestLauncherJS %>"; |       import("<%= latestLauncherJS %>"); | ||||||
|  |       window.latestJS = true; | ||||||
|     </script> |     </script> | ||||||
|  |  | ||||||
|     <script nomodule> |     <script> | ||||||
|       (function() { |       if (!window.latestJS) { | ||||||
|         // // Safari 10.1 supports type=module but ignores nomodule, so we add this check. |         <% if (useRollup) { %> | ||||||
|         if (!isS101) { |           _ls("/static/js/s.min.js").onload = function() { | ||||||
|           _ls("/static/polyfills/custom-elements-es5-adapter.js"); |             System.import("<%= es5LauncherJS %>"); | ||||||
|  |           }; | ||||||
|  |         <% } else { %> | ||||||
|           _ls("<%= es5LauncherJS %>"); |           _ls("<%= es5LauncherJS %>"); | ||||||
|  |         <% } %> | ||||||
|       } |       } | ||||||
|       })(); |  | ||||||
|     </script> |     </script> | ||||||
|     <script> |     <script> | ||||||
|     (function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){ |     (function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){ | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | import "../../../src/resources/safari-14-attachshadow-patch"; | ||||||
| import "../../../src/resources/ha-style"; | import "../../../src/resources/ha-style"; | ||||||
| import "../../../src/resources/roboto"; | import "../../../src/resources/roboto"; | ||||||
| import "./layout/hc-connect"; | import "./layout/hc-connect"; | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ import { | |||||||
|   html, |   html, | ||||||
|   LitElement, |   LitElement, | ||||||
|   property, |   property, | ||||||
|  |   internalProperty, | ||||||
|   TemplateResult, |   TemplateResult, | ||||||
| } from "lit-element"; | } from "lit-element"; | ||||||
| import { CastManager } from "../../../../src/cast/cast_manager"; | import { CastManager } from "../../../../src/cast/cast_manager"; | ||||||
| @@ -28,7 +29,7 @@ import { | |||||||
|   getLovelaceCollection, |   getLovelaceCollection, | ||||||
|   LovelaceConfig, |   LovelaceConfig, | ||||||
| } from "../../../../src/data/lovelace"; | } from "../../../../src/data/lovelace"; | ||||||
| import "../../../../src/layouts/loading-screen"; | import "../../../../src/layouts/hass-loading-screen"; | ||||||
| import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config"; | import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config"; | ||||||
| import "./hc-layout"; | import "./hc-layout"; | ||||||
| import "@material/mwc-button/mwc-button"; | import "@material/mwc-button/mwc-button"; | ||||||
| @@ -41,13 +42,13 @@ class HcCast extends LitElement { | |||||||
|  |  | ||||||
|   @property() public castManager!: CastManager; |   @property() public castManager!: CastManager; | ||||||
|  |  | ||||||
|   @property() private askWrite = false; |   @internalProperty() private askWrite = false; | ||||||
|  |  | ||||||
|   @property() private lovelaceConfig?: LovelaceConfig | null; |   @internalProperty() private lovelaceConfig?: LovelaceConfig | null; | ||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     if (this.lovelaceConfig === undefined) { |     if (this.lovelaceConfig === undefined) { | ||||||
|       return html` <loading-screen></loading-screen>> `; |       return html` <hass-loading-screen no-toolbar></hass-loading-screen>> `; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const error = |     const error = | ||||||
|   | |||||||
| @@ -17,8 +17,8 @@ import { | |||||||
|   customElement, |   customElement, | ||||||
|   html, |   html, | ||||||
|   LitElement, |   LitElement, | ||||||
|   property, |  | ||||||
|   TemplateResult, |   TemplateResult, | ||||||
|  |   internalProperty, | ||||||
| } from "lit-element"; | } from "lit-element"; | ||||||
| import { CastManager, getCastManager } from "../../../../src/cast/cast_manager"; | import { CastManager, getCastManager } from "../../../../src/cast/cast_manager"; | ||||||
| import { castSendShowDemo } from "../../../../src/cast/receiver_messages"; | import { castSendShowDemo } from "../../../../src/cast/receiver_messages"; | ||||||
| @@ -27,7 +27,7 @@ import { | |||||||
|   saveTokens, |   saveTokens, | ||||||
| } from "../../../../src/common/auth/token_storage"; | } from "../../../../src/common/auth/token_storage"; | ||||||
| import "../../../../src/components/ha-icon"; | import "../../../../src/components/ha-icon"; | ||||||
| import "../../../../src/layouts/loading-screen"; | import "../../../../src/layouts/hass-loading-screen"; | ||||||
| import { registerServiceWorker } from "../../../../src/util/register-service-worker"; | import { registerServiceWorker } from "../../../../src/util/register-service-worker"; | ||||||
| import "./hc-layout"; | import "./hc-layout"; | ||||||
|  |  | ||||||
| @@ -60,19 +60,19 @@ const INTRO = html` | |||||||
|  |  | ||||||
| @customElement("hc-connect") | @customElement("hc-connect") | ||||||
| export class HcConnect extends LitElement { | export class HcConnect extends LitElement { | ||||||
|   @property() private loading = false; |   @internalProperty() private loading = false; | ||||||
|  |  | ||||||
|   // If we had stored credentials but we cannot connect, |   // If we had stored credentials but we cannot connect, | ||||||
|   // show a screen asking retry or logout. |   // show a screen asking retry or logout. | ||||||
|   @property() private cannotConnect = false; |   @internalProperty() private cannotConnect = false; | ||||||
|  |  | ||||||
|   @property() private error?: string | TemplateResult; |   @internalProperty() private error?: string | TemplateResult; | ||||||
|  |  | ||||||
|   @property() private auth?: Auth; |   @internalProperty() private auth?: Auth; | ||||||
|  |  | ||||||
|   @property() private connection?: Connection; |   @internalProperty() private connection?: Connection; | ||||||
|  |  | ||||||
|   @property() private castManager?: CastManager | null; |   @internalProperty() private castManager?: CastManager | null; | ||||||
|  |  | ||||||
|   private openDemo = false; |   private openDemo = false; | ||||||
|  |  | ||||||
| @@ -98,7 +98,7 @@ export class HcConnect extends LitElement { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (this.castManager === undefined || this.loading) { |     if (this.castManager === undefined || this.loading) { | ||||||
|       return html` <loading-screen></loading-screen> `; |       return html` <hass-loading-screen no-toolbar></hass-loading-screen> `; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (this.castManager === null) { |     if (this.castManager === null) { | ||||||
|   | |||||||
| @@ -30,7 +30,7 @@ class HcLayout extends LitElement { | |||||||
|       <ha-card> |       <ha-card> | ||||||
|         <div class="layout"> |         <div class="layout"> | ||||||
|           <img class="hero" src="/images/google-nest-hub.png" /> |           <img class="hero" src="/images/google-nest-hub.png" /> | ||||||
|           <div class="card-header"> |           <h1 class="card-header"> | ||||||
|             Home Assistant Cast${this.subtitle ? ` – ${this.subtitle}` : ""} |             Home Assistant Cast${this.subtitle ? ` – ${this.subtitle}` : ""} | ||||||
|             ${this.auth |             ${this.auth | ||||||
|               ? html` |               ? html` | ||||||
| @@ -44,7 +44,7 @@ class HcLayout extends LitElement { | |||||||
|                   </div> |                   </div> | ||||||
|                 ` |                 ` | ||||||
|               : ""} |               : ""} | ||||||
|           </div> |           </h1> | ||||||
|           <slot></slot> |           <slot></slot> | ||||||
|         </div> |         </div> | ||||||
|       </ha-card> |       </ha-card> | ||||||
|   | |||||||
| @@ -6,13 +6,60 @@ import { castContext } from "./cast_context"; | |||||||
| import { HcMain } from "./layout/hc-main"; | import { HcMain } from "./layout/hc-main"; | ||||||
| import { ReceivedMessage } from "./types"; | import { ReceivedMessage } from "./types"; | ||||||
|  |  | ||||||
| const controller = new HcMain(); | const lovelaceController = new HcMain(); | ||||||
| document.body.append(controller); | document.body.append(lovelaceController); | ||||||
|  |  | ||||||
|  | const mediaPlayer = document.createElement("cast-media-player"); | ||||||
|  | mediaPlayer.style.display = "none"; | ||||||
|  | document.body.append(mediaPlayer); | ||||||
|  | const playerStylesAdded = false; | ||||||
|  |  | ||||||
|  | let controls: HTMLElement | null; | ||||||
|  |  | ||||||
|  | const setTouchControlsVisibility = (visible: boolean) => { | ||||||
|  |   if (!castContext.getDeviceCapabilities().touch_input_supported) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |   controls = | ||||||
|  |     controls || | ||||||
|  |     (document.body.querySelector("touch-controls") as HTMLElement | null); | ||||||
|  |   if (controls) { | ||||||
|  |     controls.style.display = visible ? "initial" : "none"; | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const showLovelaceController = () => { | ||||||
|  |   mediaPlayer.style.display = "none"; | ||||||
|  |   lovelaceController.style.display = "initial"; | ||||||
|  |   document.body.setAttribute("style", "overflow-y: auto !important"); | ||||||
|  |   setTouchControlsVisibility(false); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const showMediaPlayer = () => { | ||||||
|  |   lovelaceController.style.display = "none"; | ||||||
|  |   mediaPlayer.style.display = "initial"; | ||||||
|  |   document.body.removeAttribute("style"); | ||||||
|  |   setTouchControlsVisibility(true); | ||||||
|  |   if (!playerStylesAdded) { | ||||||
|  |     const style = document.createElement("style"); | ||||||
|  |     style.innerHTML = ` | ||||||
|  |     body { | ||||||
|  |       --logo-image: url('https://www.home-assistant.io/images/home-assistant-logo.svg'); | ||||||
|  |       --logo-repeat: no-repeat; | ||||||
|  |       --playback-logo-image: url('https://www.home-assistant.io/images/home-assistant-logo.svg'); | ||||||
|  |       --theme-hue: 200; | ||||||
|  |       --progress-color: #03a9f4; | ||||||
|  |       --splash-image: url('https://home-assistant.io/images/cast/splash.png'); | ||||||
|  |       --splash-size: cover; | ||||||
|  |     } | ||||||
|  |     `; | ||||||
|  |     document.head.appendChild(style); | ||||||
|  |   } | ||||||
|  | }; | ||||||
|  |  | ||||||
| const options = new cast.framework.CastReceiverOptions(); | const options = new cast.framework.CastReceiverOptions(); | ||||||
| options.disableIdleTimeout = true; | options.disableIdleTimeout = true; | ||||||
| options.customNamespaces = { | options.customNamespaces = { | ||||||
|   // @ts-ignore |  | ||||||
|   [CAST_NS]: cast.framework.system.MessageType.JSON, |   [CAST_NS]: cast.framework.system.MessageType.JSON, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| @@ -30,13 +77,61 @@ options.uiConfig = new cast.framework.ui.UiConfig(); | |||||||
| // @ts-ignore | // @ts-ignore | ||||||
| options.uiConfig.touchScreenOptimizedApp = true; | options.uiConfig.touchScreenOptimizedApp = true; | ||||||
|  |  | ||||||
|  | castContext.setInactivityTimeout(86400); // 1 day | ||||||
|  |  | ||||||
| castContext.addCustomMessageListener( | castContext.addCustomMessageListener( | ||||||
|   CAST_NS, |   CAST_NS, | ||||||
|   // @ts-ignore |   // @ts-ignore | ||||||
|   (ev: ReceivedMessage<HassMessage>) => { |   (ev: ReceivedMessage<HassMessage>) => { | ||||||
|  |     // We received a show Lovelace command, stop media from playing, hide media player and show Lovelace controller | ||||||
|  |     if ( | ||||||
|  |       playerManager.getPlayerState() !== | ||||||
|  |       cast.framework.messages.PlayerState.IDLE | ||||||
|  |     ) { | ||||||
|  |       playerManager.stop(); | ||||||
|  |     } else { | ||||||
|  |       showLovelaceController(); | ||||||
|  |     } | ||||||
|     const msg = ev.data; |     const msg = ev.data; | ||||||
|     msg.senderId = ev.senderId; |     msg.senderId = ev.senderId; | ||||||
|     controller.processIncomingMessage(msg); |     lovelaceController.processIncomingMessage(msg); | ||||||
|  |   } | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | const playerManager = castContext.getPlayerManager(); | ||||||
|  |  | ||||||
|  | playerManager.setMessageInterceptor( | ||||||
|  |   cast.framework.messages.MessageType.LOAD, | ||||||
|  |   (loadRequestData) => { | ||||||
|  |     // We received a play media command, hide Lovelace and show media player | ||||||
|  |     showMediaPlayer(); | ||||||
|  |     const media = loadRequestData.media; | ||||||
|  |     // Special handling if it came from Google Assistant | ||||||
|  |     if (media.entity) { | ||||||
|  |       media.contentId = media.entity; | ||||||
|  |       media.streamType = cast.framework.messages.StreamType.LIVE; | ||||||
|  |       media.contentType = "application/vnd.apple.mpegurl"; | ||||||
|  |       // @ts-ignore | ||||||
|  |       media.hlsVideoSegmentFormat = | ||||||
|  |         cast.framework.messages.HlsVideoSegmentFormat.FMP4; | ||||||
|  |     } | ||||||
|  |     return loadRequestData; | ||||||
|  |   } | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | playerManager.addEventListener( | ||||||
|  |   cast.framework.events.EventType.MEDIA_STATUS, | ||||||
|  |   (event) => { | ||||||
|  |     if ( | ||||||
|  |       event.mediaStatus?.playerState === | ||||||
|  |         cast.framework.messages.PlayerState.IDLE && | ||||||
|  |       event.mediaStatus?.idleReason && | ||||||
|  |       event.mediaStatus?.idleReason !== | ||||||
|  |         cast.framework.messages.IdleReason.INTERRUPTED | ||||||
|  |     ) { | ||||||
|  |       // media finished or stopped, return to default Lovelace | ||||||
|  |       showLovelaceController(); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| ); | ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,4 +1,10 @@ | |||||||
| import { customElement, html, property, TemplateResult } from "lit-element"; | import { | ||||||
|  |   customElement, | ||||||
|  |   html, | ||||||
|  |   property, | ||||||
|  |   internalProperty, | ||||||
|  |   TemplateResult, | ||||||
|  | } from "lit-element"; | ||||||
| import { mockHistory } from "../../../../demo/src/stubs/history"; | import { mockHistory } from "../../../../demo/src/stubs/history"; | ||||||
| import { LovelaceConfig } from "../../../../src/data/lovelace"; | import { LovelaceConfig } from "../../../../src/data/lovelace"; | ||||||
| import { | import { | ||||||
| @@ -13,9 +19,9 @@ import "./hc-lovelace"; | |||||||
|  |  | ||||||
| @customElement("hc-demo") | @customElement("hc-demo") | ||||||
| class HcDemo extends HassElement { | class HcDemo extends HassElement { | ||||||
|   @property() public lovelacePath!: string; |   @property({ attribute: false }) public lovelacePath!: string; | ||||||
|  |  | ||||||
|   @property() private _lovelaceConfig?: LovelaceConfig; |   @internalProperty() private _lovelaceConfig?: LovelaceConfig; | ||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     if (!this._lovelaceConfig) { |     if (!this._lovelaceConfig) { | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ import { HomeAssistant } from "../../../../src/types"; | |||||||
|  |  | ||||||
| @customElement("hc-launch-screen") | @customElement("hc-launch-screen") | ||||||
| class HcLaunchScreen extends LitElement { | class HcLaunchScreen extends LitElement { | ||||||
|   @property() public hass?: HomeAssistant; |   @property({ attribute: false }) public hass?: HomeAssistant; | ||||||
|  |  | ||||||
|   @property() public error?: string; |   @property() public error?: string; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -9,19 +9,20 @@ import { | |||||||
| } from "lit-element"; | } from "lit-element"; | ||||||
| import { LovelaceConfig } from "../../../../src/data/lovelace"; | import { LovelaceConfig } from "../../../../src/data/lovelace"; | ||||||
| import { Lovelace } from "../../../../src/panels/lovelace/types"; | import { Lovelace } from "../../../../src/panels/lovelace/types"; | ||||||
| import "../../../../src/panels/lovelace/views/hui-panel-view"; |  | ||||||
| import "../../../../src/panels/lovelace/views/hui-view"; | import "../../../../src/panels/lovelace/views/hui-view"; | ||||||
| import { HomeAssistant } from "../../../../src/types"; | import { HomeAssistant } from "../../../../src/types"; | ||||||
| import "./hc-launch-screen"; | import "./hc-launch-screen"; | ||||||
|  |  | ||||||
| @customElement("hc-lovelace") | @customElement("hc-lovelace") | ||||||
| class HcLovelace extends LitElement { | class HcLovelace extends LitElement { | ||||||
|   @property() public hass!: HomeAssistant; |   @property({ attribute: false }) public hass!: HomeAssistant; | ||||||
|  |  | ||||||
|   @property() public lovelaceConfig!: LovelaceConfig; |   @property({ attribute: false }) public lovelaceConfig!: LovelaceConfig; | ||||||
|  |  | ||||||
|   @property() public viewPath?: string | number; |   @property() public viewPath?: string | number; | ||||||
|  |  | ||||||
|  |   public urlPath?: string | null; | ||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     const index = this._viewIndex; |     const index = this._viewIndex; | ||||||
|     if (index === undefined) { |     if (index === undefined) { | ||||||
| @@ -35,6 +36,7 @@ class HcLovelace extends LitElement { | |||||||
|     const lovelace: Lovelace = { |     const lovelace: Lovelace = { | ||||||
|       config: this.lovelaceConfig, |       config: this.lovelaceConfig, | ||||||
|       editMode: false, |       editMode: false, | ||||||
|  |       urlPath: this.urlPath!, | ||||||
|       enableFullEditMode: () => undefined, |       enableFullEditMode: () => undefined, | ||||||
|       mode: "storage", |       mode: "storage", | ||||||
|       language: "en", |       language: "en", | ||||||
| @@ -42,20 +44,11 @@ class HcLovelace extends LitElement { | |||||||
|       deleteConfig: async () => undefined, |       deleteConfig: async () => undefined, | ||||||
|       setEditMode: () => undefined, |       setEditMode: () => undefined, | ||||||
|     }; |     }; | ||||||
|     return this.lovelaceConfig.views[index].panel |     return html` | ||||||
|       ? html` |  | ||||||
|           <hui-panel-view |  | ||||||
|             .hass=${this.hass} |  | ||||||
|             .lovelace=${lovelace} |  | ||||||
|             .config=${this.lovelaceConfig.views[index]} |  | ||||||
|           ></hui-panel-view> |  | ||||||
|         ` |  | ||||||
|       : html` |  | ||||||
|       <hui-view |       <hui-view | ||||||
|         .hass=${this.hass} |         .hass=${this.hass} | ||||||
|         .lovelace=${lovelace} |         .lovelace=${lovelace} | ||||||
|         .index=${index} |         .index=${index} | ||||||
|             columns="2" |  | ||||||
|       ></hui-view> |       ></hui-view> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| @@ -73,7 +66,7 @@ class HcLovelace extends LitElement { | |||||||
|  |  | ||||||
|         if (configBackground) { |         if (configBackground) { | ||||||
|           (this.shadowRoot!.querySelector( |           (this.shadowRoot!.querySelector( | ||||||
|             "hui-view, hui-panel-view" |             "hui-view" | ||||||
|           ) as HTMLElement)!.style.setProperty( |           ) as HTMLElement)!.style.setProperty( | ||||||
|             "--lovelace-background", |             "--lovelace-background", | ||||||
|             configBackground |             configBackground | ||||||
|   | |||||||
| @@ -3,7 +3,12 @@ import { | |||||||
|   getAuth, |   getAuth, | ||||||
|   UnsubscribeFunc, |   UnsubscribeFunc, | ||||||
| } from "home-assistant-js-websocket"; | } from "home-assistant-js-websocket"; | ||||||
| import { customElement, html, property, TemplateResult } from "lit-element"; | import { | ||||||
|  |   customElement, | ||||||
|  |   html, | ||||||
|  |   internalProperty, | ||||||
|  |   TemplateResult, | ||||||
|  | } from "lit-element"; | ||||||
| import { CAST_NS } from "../../../../src/cast/const"; | import { CAST_NS } from "../../../../src/cast/const"; | ||||||
| import { | import { | ||||||
|   ConnectMessage, |   ConnectMessage, | ||||||
| @@ -31,13 +36,13 @@ let resourcesLoaded = false; | |||||||
|  |  | ||||||
| @customElement("hc-main") | @customElement("hc-main") | ||||||
| export class HcMain extends HassElement { | export class HcMain extends HassElement { | ||||||
|   @property() private _showDemo = false; |   @internalProperty() private _showDemo = false; | ||||||
|  |  | ||||||
|   @property() private _lovelaceConfig?: LovelaceConfig; |   @internalProperty() private _lovelaceConfig?: LovelaceConfig; | ||||||
|  |  | ||||||
|   @property() private _lovelacePath: string | number | null = null; |   @internalProperty() private _lovelacePath: string | number | null = null; | ||||||
|  |  | ||||||
|   @property() private _error?: string; |   @internalProperty() private _error?: string; | ||||||
|  |  | ||||||
|   private _unsubLovelace?: UnsubscribeFunc; |   private _unsubLovelace?: UnsubscribeFunc; | ||||||
|  |  | ||||||
| @@ -82,6 +87,8 @@ export class HcMain extends HassElement { | |||||||
|         .hass=${this.hass} |         .hass=${this.hass} | ||||||
|         .lovelaceConfig=${this._lovelaceConfig} |         .lovelaceConfig=${this._lovelaceConfig} | ||||||
|         .viewPath=${this._lovelacePath} |         .viewPath=${this._lovelacePath} | ||||||
|  |         .urlPath=${this._urlPath} | ||||||
|  |         @config-refresh=${this._generateLovelaceConfig} | ||||||
|       ></hc-lovelace> |       ></hc-lovelace> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| @@ -191,14 +198,11 @@ export class HcMain extends HassElement { | |||||||
|           this._handleNewLovelaceConfig(lovelaceConfig) |           this._handleNewLovelaceConfig(lovelaceConfig) | ||||||
|         ); |         ); | ||||||
|       } catch (err) { |       } catch (err) { | ||||||
|  |         // eslint-disable-next-line | ||||||
|  |         console.log("Error fetching Lovelace configuration", err, msg); | ||||||
|         // Generate a Lovelace config. |         // Generate a Lovelace config. | ||||||
|         this._unsubLovelace = () => undefined; |         this._unsubLovelace = () => undefined; | ||||||
|         const { generateLovelaceConfigFromHass } = await import( |         await this._generateLovelaceConfig(); | ||||||
|           "../../../../src/panels/lovelace/common/generate-lovelace-config" |  | ||||||
|         ); |  | ||||||
|         this._handleNewLovelaceConfig( |  | ||||||
|           await generateLovelaceConfigFromHass(this.hass!) |  | ||||||
|         ); |  | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     if (!resourcesLoaded) { |     if (!resourcesLoaded) { | ||||||
| @@ -212,12 +216,19 @@ export class HcMain extends HassElement { | |||||||
|     } |     } | ||||||
|     this._showDemo = false; |     this._showDemo = false; | ||||||
|     this._lovelacePath = msg.viewPath; |     this._lovelacePath = msg.viewPath; | ||||||
|     if (castContext.getDeviceCapabilities().touch_input_supported) { |  | ||||||
|       this._breakFree(); |  | ||||||
|     } |  | ||||||
|     this._sendStatus(); |     this._sendStatus(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private async _generateLovelaceConfig() { | ||||||
|  |     const { generateLovelaceConfigFromHass } = await import( | ||||||
|  |       "../../../../src/panels/lovelace/common/generate-lovelace-config" | ||||||
|  |     ); | ||||||
|  |     this._handleNewLovelaceConfig( | ||||||
|  |       await generateLovelaceConfigFromHass(this.hass!) | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private _handleNewLovelaceConfig(lovelaceConfig: LovelaceConfig) { |   private _handleNewLovelaceConfig(lovelaceConfig: LovelaceConfig) { | ||||||
|     castContext.setApplicationState(lovelaceConfig.title!); |     castContext.setApplicationState(lovelaceConfig.title!); | ||||||
|     this._lovelaceConfig = lovelaceConfig; |     this._lovelaceConfig = lovelaceConfig; | ||||||
| @@ -228,9 +239,6 @@ export class HcMain extends HassElement { | |||||||
|       this._showDemo = true; |       this._showDemo = true; | ||||||
|       this._lovelacePath = "overview"; |       this._lovelacePath = "overview"; | ||||||
|       this._sendStatus(); |       this._sendStatus(); | ||||||
|       if (castContext.getDeviceCapabilities().touch_input_supported) { |  | ||||||
|         this._breakFree(); |  | ||||||
|       } |  | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -251,14 +259,6 @@ export class HcMain extends HassElement { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _breakFree() { |  | ||||||
|     const controls = document.body.querySelector("touch-controls"); |  | ||||||
|     if (controls) { |  | ||||||
|       controls.remove(); |  | ||||||
|     } |  | ||||||
|     document.body.setAttribute("style", "overflow-y: auto !important"); |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   private sendMessage(senderId: string, response: any) { |   private sendMessage(senderId: string, response: any) { | ||||||
|     castContext.sendCustomMessage(CAST_NS, senderId, response); |     castContext.sendCustomMessage(CAST_NS, senderId, response); | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
| import "web-animations-js/web-animations-next-lite.min"; | import "web-animations-js/web-animations-next-lite.min"; | ||||||
| import "../../../src/resources/roboto"; | import "../../../src/resources/roboto"; | ||||||
|  | import "../../../src/resources/ha-style"; | ||||||
| import "./layout/hc-lovelace"; | import "./layout/hc-lovelace"; | ||||||
|   | |||||||
| @@ -1,11 +1,8 @@ | |||||||
| const { createCastConfig } = require("../build-scripts/webpack.js"); | const { createCastConfig } = require("../build-scripts/webpack.js"); | ||||||
| const { isProdBuild } = require("../build-scripts/env.js"); | const { isProdBuild, isStatsBuild } = require("../build-scripts/env.js"); | ||||||
|  |  | ||||||
| // File just used for stats builds |  | ||||||
|  |  | ||||||
| const latestBuild = true; |  | ||||||
|  |  | ||||||
| module.exports = createCastConfig({ | module.exports = createCastConfig({ | ||||||
|   isProdBuild: isProdBuild(), |   isProdBuild: isProdBuild(), | ||||||
|   latestBuild, |   isStatsBuild: isStatsBuild(), | ||||||
|  |   latestBuild: true, | ||||||
| }); | }); | ||||||
|   | |||||||
| Before Width: | Height: | Size: 1.8 KiB After Width: | Height: | Size: 1.5 KiB | 
| Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 19 KiB | 
| Before Width: | Height: | Size: 20 KiB After Width: | Height: | Size: 20 KiB | 
| Before Width: | Height: | Size: 3.3 KiB After Width: | Height: | Size: 2.9 KiB | 
| Before Width: | Height: | Size: 8.4 KiB After Width: | Height: | Size: 7.2 KiB | 
| Before Width: | Height: | Size: 3.2 KiB After Width: | Height: | Size: 2.7 KiB | 
| Before Width: | Height: | Size: 5.7 KiB After Width: | Height: | Size: 5.2 KiB | 
| Before Width: | Height: | Size: 3.1 KiB After Width: | Height: | Size: 2.6 KiB | 
| Before Width: | Height: | Size: 7.3 KiB After Width: | Height: | Size: 7.1 KiB | 
| Before Width: | Height: | Size: 6.1 KiB After Width: | Height: | Size: 5.9 KiB | 
| Before Width: | Height: | Size: 7.7 KiB After Width: | Height: | Size: 7.6 KiB | 
| Before Width: | Height: | Size: 6.4 KiB After Width: | Height: | Size: 6.2 KiB | 
| Before Width: | Height: | Size: 6.5 KiB After Width: | Height: | Size: 6.2 KiB | 
| Before Width: | Height: | Size: 9.9 KiB After Width: | Height: | Size: 8.7 KiB | 
| Before Width: | Height: | Size: 7.9 KiB After Width: | Height: | Size: 7.7 KiB | 
| Before Width: | Height: | Size: 6.5 KiB After Width: | Height: | Size: 6.3 KiB | 
| Before Width: | Height: | Size: 1.8 KiB After Width: | Height: | Size: 1.5 KiB | 
| Before Width: | Height: | Size: 4.6 KiB After Width: | Height: | Size: 3.9 KiB | 
| Before Width: | Height: | Size: 4.5 KiB After Width: | Height: | Size: 3.9 KiB | 
| Before Width: | Height: | Size: 5.5 KiB After Width: | Height: | Size: 5.2 KiB | 
| Before Width: | Height: | Size: 8.2 KiB After Width: | Height: | Size: 7.9 KiB | 
| Before Width: | Height: | Size: 8.5 KiB After Width: | Height: | Size: 7.2 KiB | 
| Before Width: | Height: | Size: 11 KiB After Width: | Height: | Size: 9.6 KiB | 
| Before Width: | Height: | Size: 3.7 KiB After Width: | Height: | Size: 3.1 KiB | 
| Before Width: | Height: | Size: 3.3 KiB After Width: | Height: | Size: 2.9 KiB | 
| Before Width: | Height: | Size: 9.9 KiB After Width: | Height: | Size: 8.7 KiB | 
| Before Width: | Height: | Size: 4.9 KiB After Width: | Height: | Size: 4.4 KiB | 
| Before Width: | Height: | Size: 767 B After Width: | Height: | Size: 532 B | 
| Before Width: | Height: | Size: 803 B After Width: | Height: | Size: 535 B | 
| Before Width: | Height: | Size: 75 KiB After Width: | Height: | Size: 75 KiB | 
| Before Width: | Height: | Size: 56 KiB After Width: | Height: | Size: 56 KiB | 
| Before Width: | Height: | Size: 58 KiB After Width: | Height: | Size: 58 KiB | 
| Before Width: | Height: | Size: 71 KiB After Width: | Height: | Size: 71 KiB | 
| Before Width: | Height: | Size: 8.2 KiB After Width: | Height: | Size: 7.1 KiB | 
| Before Width: | Height: | Size: 33 KiB After Width: | Height: | Size: 32 KiB | 
| Before Width: | Height: | Size: 106 KiB After Width: | Height: | Size: 94 KiB | 
| Before Width: | Height: | Size: 60 KiB After Width: | Height: | Size: 60 KiB |