mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-26 12:09:47 +00:00 
			
		
		
		
	Compare commits
	
		
			595 Commits
		
	
	
		
			20220222.0
			...
			fix-button
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 026d027386 | ||
|   | d0ead1fdb8 | ||
|   | b0e6c41238 | ||
|   | 2c1550b10f | ||
|   | ffc4ca5b56 | ||
|   | 85ad6619b7 | ||
|   | 7358faf88e | ||
|   | 19d014307a | ||
|   | 5217f5c50c | ||
|   | c4624faa71 | ||
|   | b35ba4d673 | ||
|   | f8303bff76 | ||
|   | e61aa266a6 | ||
|   | d7971c69ad | ||
|   | 966a624ef6 | ||
|   | 7cc576a616 | ||
|   | 2dec8e70ec | ||
|   | 97663aef42 | ||
|   | 3f1a2526b3 | ||
|   | e7517a8b61 | ||
|   | e3d394eb32 | ||
|   | 536ea822b3 | ||
|   | 8e4e22b6f8 | ||
|   | 2eaa246a03 | ||
|   | e841bf89be | ||
|   | 36e1203fb1 | ||
|   | 3acab5a39c | ||
|   | 49cfde1fe7 | ||
|   | 49c018c000 | ||
|   | b71b230bfd | ||
|   | e1fd7244a5 | ||
|   | 067c2fdfa8 | ||
|   | a02b817d7f | ||
|   | 7db6e0b779 | ||
|   | 1d5cc91a2d | ||
|   | 0623e7dce4 | ||
|   | da106d278c | ||
|   | 51c5ab33f0 | ||
|   | 8ac4a6d900 | ||
|   | fae1bcf0e0 | ||
|   | 9a9eec40b2 | ||
|   | 6ab19d66d5 | ||
|   | a0a7ce014f | ||
|   | bfeb90780f | ||
|   | 1f105b6c15 | ||
|   | 5b7b0ea326 | ||
|   | 32a991989f | ||
|   | 788f76ab9c | ||
|   | f6411dce66 | ||
|   | 6f19ea1d84 | ||
|   | 448609533f | ||
|   | 6c48ace41e | ||
|   | c41e100c1c | ||
|   | 8216b522c2 | ||
|   | 82035d587a | ||
|   | 2796c3570a | ||
|   | f4f51e1de5 | ||
|   | af6b0d3266 | ||
|   | 7d1c77a38f | ||
|   | f807618f75 | ||
|   | 4cfb6713cb | ||
|   | d32f84f28d | ||
|   | 5fb1504211 | ||
|   | c37e1f0c9d | ||
|   | 90c234ffad | ||
|   | dd3a3ec586 | ||
|   | 6f67da09c0 | ||
|   | ba27c184f6 | ||
|   | b37f97128a | ||
|   | ee0de942f7 | ||
|   | ae2d48f2f4 | ||
|   | 1bd760b455 | ||
|   | 3d66a68791 | ||
|   | 01a53439c4 | ||
|   | 09ee8dbeb6 | ||
|   | f36c91550d | ||
|   | 6be6c711d0 | ||
|   | 72a36fb1cd | ||
|   | 4c982b3323 | ||
|   | c9c3be71cc | ||
|   | f1b965dcc5 | ||
|   | a08a23a93d | ||
|   | 2040a49458 | ||
|   | df94f4f907 | ||
|   | 96d375cb84 | ||
|   | 7a9c2f56c5 | ||
|   | 5ec7193e5c | ||
|   | d89e4337f2 | ||
|   | 2e192d5021 | ||
|   | 7db28c0156 | ||
|   | f09c842981 | ||
|   | b295bbd706 | ||
|   | 8d3132fefc | ||
|   | 00c5d3dbbb | ||
|   | ca37aff47d | ||
|   | 9ed069ef6a | ||
|   | 6faa3eb848 | ||
|   | 6c73ae5bf7 | ||
|   | ce77ddf365 | ||
|   | cf05fbaa9d | ||
|   | 552c474feb | ||
|   | a4f8e886bc | ||
|   | cc0c96b8b4 | ||
|   | 445f0e23fe | ||
|   | 6f240297d1 | ||
|   | 6da4981b70 | ||
|   | cfadf4d700 | ||
|   | 7e60de0531 | ||
|   | aaef6d7b91 | ||
|   | 58c5ce2638 | ||
|   | a9d01c7b55 | ||
|   | c5de8a4361 | ||
|   | b53645ce92 | ||
|   | de34a5a597 | ||
|   | bd8e15bdd1 | ||
|   | 45c7e0eeeb | ||
|   | a35a380ec7 | ||
|   | 02e67d1146 | ||
|   | a5411f7ac4 | ||
|   | e8da203fe1 | ||
|   | 10aa0a8829 | ||
|   | 85a37e2d2f | ||
|   | ba8621fa2c | ||
|   | 43e80f1a2e | ||
|   | 3a305a44b6 | ||
|   | e99143139e | ||
|   | f0c7232704 | ||
|   | b2186592df | ||
|   | e51e3e79d5 | ||
|   | 3b6b4d7664 | ||
|   | 239e71b414 | ||
|   | 080cad0ccd | ||
|   | dd49fd2788 | ||
|   | a571fb5528 | ||
|   | 1369c1ae8c | ||
|   | f5864181af | ||
|   | a4a0d7cf19 | ||
|   | 092dfd1e87 | ||
|   | a29ac33810 | ||
|   | 1421df2a5a | ||
|   | 591b8cc503 | ||
|   | 011467ece0 | ||
|   | f52e8c3392 | ||
|   | c8b87b65bd | ||
|   | 98cc82db44 | ||
|   | f510e2a8e0 | ||
|   | 3438912ba5 | ||
|   | 671c8e387f | ||
|   | 0108ec65cf | ||
|   | 39f7034578 | ||
|   | bf8affaf2b | ||
|   | e16a61eb53 | ||
|   | cadbe45bab | ||
|   | 51f971337d | ||
|   | 1f3c23de29 | ||
|   | bdfb17d957 | ||
|   | 8c97aee1fe | ||
|   | 38b4090daa | ||
|   | b8c55f2f65 | ||
|   | 7ca379e0a1 | ||
|   | 1617a9dfed | ||
|   | 2c9411c6c3 | ||
|   | 67626d4a06 | ||
|   | 8135611688 | ||
|   | 3ccbf6983e | ||
|   | e4f91195d8 | ||
|   | 2751f8f33b | ||
|   | 57f2df3b3e | ||
|   | 6822f0d067 | ||
|   | cfba957313 | ||
|   | 3149ffbf19 | ||
|   | 4cd8b76d7e | ||
|   | 4b644d8bc5 | ||
|   | 307cd5ad8c | ||
|   | ebc807a6a4 | ||
|   | 66adecdfc9 | ||
|   | 2cc6432a0f | ||
|   | a2c0c0474a | ||
|   | 27884b9a54 | ||
|   | 293df61872 | ||
|   | f82dada3e5 | ||
|   | e5824c4794 | ||
|   | 186550229c | ||
|   | 7877dd8e6b | ||
|   | b03abc249b | ||
|   | fda03918b9 | ||
|   | 6747375a1b | ||
|   | 53b6e31881 | ||
|   | fa004de2d1 | ||
|   | 3605f7b70f | ||
|   | 5348c54c91 | ||
|   | 684e4421bc | ||
|   | 28f5611df5 | ||
|   | 8da73d49d7 | ||
|   | 049ddd5f84 | ||
|   | 8ae2d4e93a | ||
|   | 824bb9ba35 | ||
|   | d550b1a18e | ||
|   | dea6c0e761 | ||
|   | 9caee357c0 | ||
|   | 35d892c418 | ||
|   | 9572a2a46b | ||
|   | 8996361b26 | ||
|   | 02ee731602 | ||
|   | bb1e6bf35b | ||
|   | c1b65285c1 | ||
|   | 8b8d6e5fa3 | ||
|   | c34fe184e8 | ||
|   | 7363838f86 | ||
|   | 3081425ccd | ||
|   | 95d494a54c | ||
|   | 145e5d7bc6 | ||
|   | 876fd9e85a | ||
|   | e8c30cabca | ||
|   | 490f84a7b1 | ||
|   | ca28178b86 | ||
|   | 2fceb0aeee | ||
|   | 86f39d1d43 | ||
|   | 1faf60444d | ||
|   | e927091d21 | ||
|   | cff2f856b3 | ||
|   | a743e3bbba | ||
|   | f8a52d250e | ||
|   | b70a523bdf | ||
|   | 8f2ed747e6 | ||
|   | 5deccefb15 | ||
|   | 3f04abfa9d | ||
|   | 8e55c83996 | ||
|   | dee59486ba | ||
|   | 77ef509aea | ||
|   | bfa7bccfa6 | ||
|   | a8c365edc8 | ||
|   | 94953ddf6c | ||
|   | 6b67546daf | ||
|   | 3e188d1f87 | ||
|   | f69eb15a90 | ||
|   | dfe348187f | ||
|   | 9706c56c5c | ||
|   | 3677c5be2c | ||
|   | bd339fa963 | ||
|   | 28f1b6bdf4 | ||
|   | c5aac3b81d | ||
|   | 70836597e9 | ||
|   | 958a1de2fd | ||
|   | 36d30266e3 | ||
|   | 558ab9761d | ||
|   | 269ef370e4 | ||
|   | ba2958ecd2 | ||
|   | 3b8b6eb315 | ||
|   | 4f13db3178 | ||
|   | ee7aa54ab4 | ||
|   | c305dd4cd5 | ||
|   | 6865791596 | ||
|   | 2099259393 | ||
|   | 27ca45dc70 | ||
|   | d290c11219 | ||
|   | cabe10ffdb | ||
|   | aa562c21a8 | ||
|   | 22175a7271 | ||
|   | 1e0647c0d1 | ||
|   | 58d94da8b3 | ||
|   | d97763a3e8 | ||
|   | aa129aa123 | ||
|   | f648317206 | ||
|   | 0685fdf7c6 | ||
|   | 6fd4cda534 | ||
|   | 511368da13 | ||
|   | 76e1721c58 | ||
|   | bad5a389b5 | ||
|   | 85d1f49763 | ||
|   | 7723d47ac1 | ||
|   | 30b130ca74 | ||
|   | a124ec0717 | ||
|   | 323d98ecf7 | ||
|   | 125a601ae3 | ||
|   | 3c549c6b31 | ||
|   | 9c1494c74d | ||
|   | e751abd775 | ||
|   | 714f2447b7 | ||
|   | d900e40d04 | ||
|   | 8b82383790 | ||
|   | 5a2cc2646c | ||
|   | 16a0902989 | ||
|   | 8f67aa38af | ||
|   | 34184cf2ab | ||
|   | 611cd2818e | ||
|   | 0a4e8fd5d0 | ||
|   | 11f0361f48 | ||
|   | cfa048ea4e | ||
|   | bbca7b762b | ||
|   | 1dba849567 | ||
|   | aff1ec10bf | ||
|   | 351ec08a71 | ||
|   | a1a6a2cd30 | ||
|   | 4e82c23b29 | ||
|   | 59595aabde | ||
|   | 358f91c2a9 | ||
|   | e0e01e68b4 | ||
|   | 61dc4eaaea | ||
|   | 65c4d02452 | ||
|   | f78ce2c844 | ||
|   | 4d1ab83b30 | ||
|   | fb4b40b828 | ||
|   | db0c4ef941 | ||
|   | c5b60b826b | ||
|   | 718f0330a7 | ||
|   | 89e31486c5 | ||
|   | 717eec1860 | ||
|   | b6e51352e3 | ||
|   | 2ade728bc3 | ||
|   | 62f227da83 | ||
|   | 9557b604da | ||
|   | b45c355c9f | ||
|   | 0b47d2c687 | ||
|   | 8baa0b2a9b | ||
|   | c68a1d21ff | ||
|   | 419d659311 | ||
|   | ba8b20d877 | ||
|   | 8de542388f | ||
|   | e6c580aadc | ||
|   | 11696c566a | ||
|   | edc15940a2 | ||
|   | bf35ee549d | ||
|   | 4c3baa678c | ||
|   | 0bb2767696 | ||
|   | 80a4852325 | ||
|   | 9c3e0fc997 | ||
|   | 9e4bee123f | ||
|   | c2c09b1284 | ||
|   | bad776b979 | ||
|   | 396791b805 | ||
|   | 2b1457e1cd | ||
|   | b5861869e3 | ||
|   | 9444228907 | ||
|   | 86afd883a5 | ||
|   | 062f21aa91 | ||
|   | ba235ac797 | ||
|   | 505c22248b | ||
|   | 624cb48f78 | ||
|   | 7ab54ee5ce | ||
|   | f5af63a50e | ||
|   | ff80ab34ee | ||
|   | cfc1999a28 | ||
|   | 7ca28469b7 | ||
|   | ac670614b4 | ||
|   | e263b57296 | ||
|   | c7050e4676 | ||
|   | 00cbd1d9e6 | ||
|   | 2a12172eeb | ||
|   | 85d3011625 | ||
|   | ca22ec6340 | ||
|   | 61f6e8855b | ||
|   | a44b8981e1 | ||
|   | b080bca9ce | ||
|   | d30e8ee9d8 | ||
|   | 637e4203e5 | ||
|   | 2648a53bbc | ||
|   | b3fa0cccb4 | ||
|   | dd963be723 | ||
|   | 224df896a1 | ||
|   | a58b4fb262 | ||
|   | 27ca61ec85 | ||
|   | 859f49f3eb | ||
|   | 40d878689f | ||
|   | 420e8fe1ff | ||
|   | df96199433 | ||
|   | f493280f0a | ||
|   | cbd030a379 | ||
|   | 95b80accc9 | ||
|   | c522670815 | ||
|   | 7b6d3c0e36 | ||
|   | 504b043159 | ||
|   | dffc66ccc3 | ||
|   | c7e9ee785d | ||
|   | 079cc39a6e | ||
|   | d6a1d5af79 | ||
|   | c0dce08e19 | ||
|   | a7a347ed05 | ||
|   | 2d9b50defc | ||
|   | 840858b18c | ||
|   | afd2e71f6c | ||
|   | 88af0aa788 | ||
|   | 49124f6f09 | ||
|   | 73f5580555 | ||
|   | bdde5268c6 | ||
|   | 15e972c158 | ||
|   | 0fc4c24f5a | ||
|   | 9eba50df0c | ||
|   | 0e0e07437f | ||
|   | 6ac51ede52 | ||
|   | ccf1fb573a | ||
|   | fa537968c4 | ||
|   | 6bf2111a3c | ||
|   | f5f8be8276 | ||
|   | ddf1cc0733 | ||
|   | 9c1d1cb6f6 | ||
|   | 470225abde | ||
|   | ee230b86c1 | ||
|   | f927fc64a9 | ||
|   | 03677c33f7 | ||
|   | bc36a206da | ||
|   | af06ab1e2d | ||
|   | 3e2135a485 | ||
|   | 2e7f8fb46f | ||
|   | 102568c4bd | ||
|   | 4fcdae842e | ||
|   | ea19740f5a | ||
|   | 3e0942b631 | ||
|   | 0261cea796 | ||
|   | 5247b2813f | ||
|   | 8a5090684e | ||
|   | 1784ba5e68 | ||
|   | 4fbe9a7b10 | ||
|   | 1ca9c7838a | ||
|   | 4fc2c3ef05 | ||
|   | 73ff8e28a8 | ||
|   | dde1c5e03c | ||
|   | 01eed22592 | ||
|   | 94ebb63589 | ||
|   | 29119db5ce | ||
|   | 9908162ac2 | ||
|   | 1e929ae78a | ||
|   | ab5df0fe6e | ||
|   | d5010dda9e | ||
|   | 4ac097f32b | ||
|   | 5d3d15072f | ||
|   | 5c53bc4225 | ||
|   | d5a307f8f4 | ||
|   | a27dd1e7f1 | ||
|   | c86ed1fb3e | ||
|   | 7fa7a48072 | ||
|   | 4e0fc8ee08 | ||
|   | 5f6490e54e | ||
|   | db78b046a2 | ||
|   | c37fe1e7ff | ||
|   | f1ec479d41 | ||
|   | e01cb3ca82 | ||
|   | b8d3c68a7a | ||
|   | 641003bb2a | ||
|   | 3358fc2b18 | ||
|   | f9ccfa00a2 | ||
|   | 070e11a2db | ||
|   | dcf50e055b | ||
|   | fc1c6cea24 | ||
|   | 1fa04baa16 | ||
|   | c6a103bd30 | ||
|   | 84ffa2369a | ||
|   | 9d1618024e | ||
|   | 307aa161a6 | ||
|   | affa6a92e7 | ||
|   | 1e5ec241d5 | ||
|   | 27a98a32fc | ||
|   | 2c8ac58f97 | ||
|   | 6b995969b1 | ||
|   | 72c107484a | ||
|   | d1085b6657 | ||
|   | cc27ddb362 | ||
|   | c4dc6bfb0d | ||
|   | 4fbcc30a37 | ||
|   | 4916527e5f | ||
|   | fad8a27232 | ||
|   | a993d3a753 | ||
|   | 5dfe17a43a | ||
|   | 9b6c935ffb | ||
|   | f4e28da0a3 | ||
|   | 294a69d7e4 | ||
|   | f89b8cffcf | ||
|   | 99fd3a1b6f | ||
|   | 246e426182 | ||
|   | 9f1e9b43fe | ||
|   | 8301ae262c | ||
|   | d968fe41ee | ||
|   | db830e9014 | ||
|   | fc6b594a27 | ||
|   | 86dbf99ebe | ||
|   | 68e7ce1883 | ||
|   | e9003ac35e | ||
|   | 1dd5214b42 | ||
|   | 96738350bb | ||
|   | 5bdecf57cf | ||
|   | ec12282f8c | ||
|   | 552dbca201 | ||
|   | 0bbc0ebb3c | ||
|   | ac7acc5802 | ||
|   | 64e1d160d1 | ||
|   | 8e51878b6d | ||
|   | 7c94ced303 | ||
|   | a040e1d5e0 | ||
|   | 87c7407857 | ||
|   | d0d0c44ec7 | ||
|   | 4cdff3faea | ||
|   | 0dac10aa23 | ||
|   | 4b8b14a69d | ||
|   | 9d28df31bd | ||
|   | 8258641443 | ||
|   | dfcb0f6ba0 | ||
|   | 2e10eb04b6 | ||
|   | b4b52d3872 | ||
|   | 3873203721 | ||
|   | ccb91e0b49 | ||
|   | bd20c15a55 | ||
|   | 0936fd9ae4 | ||
|   | adefc7a4e2 | ||
|   | 8f8017ecff | ||
|   | 604b79696e | ||
|   | 8c445f6409 | ||
|   | 797c871137 | ||
|   | 24829bd903 | ||
|   | add92a559d | ||
|   | 7f086c0900 | ||
|   | 17018c0f26 | ||
|   | cd6a478130 | ||
|   | 4f6d7ca5c9 | ||
|   | c2994343b4 | ||
|   | e5f77c35d4 | ||
|   | a9e5a5dd44 | ||
|   | 1159798b8d | ||
|   | 437de42c55 | ||
|   | 89e0bb3f16 | ||
|   | 28c9631b6c | ||
|   | 8882624618 | ||
|   | a769f84755 | ||
|   | 7abf9c2473 | ||
|   | 298296a81f | ||
|   | 6907fa5c8e | ||
|   | 546461b70f | ||
|   | 4031009c26 | ||
|   | 91e4557625 | ||
|   | f0c4b92dbb | ||
|   | ffac3d055e | ||
|   | 04ae8c9d14 | ||
|   | 0158610d42 | ||
|   | 5ab6121581 | ||
|   | 3d9c31aef9 | ||
|   | acfeea5c92 | ||
|   | 75e8e17073 | ||
|   | 976fd4b32d | ||
|   | 49beafbe5f | ||
|   | 151f8d5524 | ||
|   | 48355aa98e | ||
|   | fc31929f41 | ||
|   | b7c149fcc1 | ||
|   | 02d058561b | ||
|   | 4e57fb1ec1 | ||
|   | 30f79c5a46 | ||
|   | 30f7252d84 | ||
|   | 8af795a7ce | ||
|   | 8576eeae41 | ||
|   | cd740ed135 | ||
|   | 892f774792 | ||
|   | aa504fe1f8 | ||
|   | be491451d5 | ||
|   | bad184210d | ||
|   | a43b3b64b3 | ||
|   | aa831a9adf | ||
|   | 43d4f55392 | ||
|   | 130c66fb24 | ||
|   | 684c232c8c | ||
|   | 09f8f816d1 | ||
|   | 1719d062b3 | ||
|   | 87290c4330 | ||
|   | fec0dc0032 | ||
|   | 70ca27c8c9 | ||
|   | 9ae1f01ad6 | ||
|   | 0113cc3cf6 | ||
|   | 2a98ace0b3 | ||
|   | 5f69a4c165 | ||
|   | 8db22d4f88 | ||
|   | 3204dbfc4d | ||
|   | 430b47fc4a | ||
|   | 5d8b3227f3 | ||
|   | b341ee9d38 | ||
|   | e6dbbc31a8 | ||
|   | 0010bf5a8f | ||
|   | 6e2e80a297 | ||
|   | aa9ff01030 | ||
|   | 7f8ecf57d7 | ||
|   | 6be6755f6f | ||
|   | 64459a06c6 | ||
|   | df35496c6e | ||
|   | aa988c758d | ||
|   | 1dd1095d19 | ||
|   | 7e68393c84 | ||
|   | 540c06c9f7 | ||
|   | f633cc2b0d | ||
|   | 1baaf76471 | ||
|   | 41ec65ef3d | ||
|   | 79e1e195a0 | ||
|   | dfbf7fb436 | ||
|   | f37a5fa021 | ||
|   | 5e2fcf928c | ||
|   | 35a41b3490 | ||
|   | f59cb661cd | ||
|   | 51938fb51f | ||
|   | c85236e251 | 
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE.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: Configuration -> Info. |   Home Assistant frontend: Settings -> About. | ||||||
|  |  | ||||||
|   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. | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| name: Report a bug with the UI, Frontend or Lovelace | name: Report a bug with the UI / Dashboards | ||||||
| description: Report an issue related to the Home Assistant frontend. | description: Report an issue related to the Home Assistant frontend. | ||||||
| labels: bug | labels: bug | ||||||
| body: | body: | ||||||
| @@ -9,7 +9,7 @@ body: | |||||||
|  |  | ||||||
|         If you have a feature or enhancement request for the frontend, please [start an discussion][fr] instead of creating an issue. |         If you have a feature or enhancement request for the frontend, please [start an discussion][fr] instead of creating an issue. | ||||||
|  |  | ||||||
|         **Please not not report issues for custom Lovelace cards.** |         **Please not not report issues for custom cards.** | ||||||
|  |  | ||||||
|         [fr]: https://github.com/home-assistant/frontend/discussions |         [fr]: https://github.com/home-assistant/frontend/discussions | ||||||
|         [releases]: https://github.com/home-assistant/home-assistant/releases |         [releases]: https://github.com/home-assistant/home-assistant/releases | ||||||
| @@ -64,7 +64,7 @@ body: | |||||||
|       label: What version of Home Assistant Core has the issue? |       label: What version of Home Assistant Core has the issue? | ||||||
|       placeholder: core- |       placeholder: core- | ||||||
|       description: > |       description: > | ||||||
|         Can be found in the Configuration panel -> Info. |         Can be found in: [Settings -> About](https://my.home-assistant.io/redirect/info/). | ||||||
|   - type: input |   - type: input | ||||||
|     attributes: |     attributes: | ||||||
|       label: What was the last working version of Home Assistant Core? |       label: What was the last working version of Home Assistant Core? | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,17 +1,17 @@ | |||||||
| blank_issues_enabled: false | blank_issues_enabled: false | ||||||
| contact_links: | contact_links: | ||||||
|   - name: Request a feature for the UI, Frontend or Lovelace |   - name: Request a feature for the UI / Dashboards | ||||||
|     url: https://github.com/home-assistant/frontend/discussions/category_choices |     url: https://github.com/home-assistant/frontend/discussions/category_choices | ||||||
|     about: Request an new feature for the Home Assistant frontend. |     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 / Dashboards | ||||||
|     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 in the backend ("core") repository. | ||||||
|   - name: Report incorrect or missing information on our website |   - name: Report incorrect or missing information on our website | ||||||
|     url: https://github.com/home-assistant/home-assistant.io/issues |     url: https://github.com/home-assistant/home-assistant.io/issues | ||||||
|     about: Our documentation has its own issue tracker. Please report issues with the website there. |     about: Our documentation has its own issue tracker. Please report issues with the website there. | ||||||
|   - name: I have a question or need support |   - name: I have a question or need support | ||||||
|     url: https://www.home-assistant.io/help |     url: https://www.home-assistant.io/help | ||||||
|     about: We use GitHub for tracking bugs, check our website for resources on getting help. |     about: We use GitHub for tracking bugs. Check our website for resources on getting help. | ||||||
|   - name: I'm unsure where to go |   - name: I'm unsure where to go | ||||||
|     url: https://www.home-assistant.io/join-chat |     url: https://www.home-assistant.io/join-chat | ||||||
|     about: If you are unsure where to go, then joining our chat is recommended; Just ask! |     about: If you are unsure where to go, then joining our chat is recommended; Just ask! | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -10,10 +10,18 @@ env: | |||||||
|   NODE_VERSION: 14 |   NODE_VERSION: 14 | ||||||
|   NODE_OPTIONS: --max_old_space_size=6144 |   NODE_OPTIONS: --max_old_space_size=6144 | ||||||
|  |  | ||||||
|  | # Set default workflow permissions | ||||||
|  | # All scopes not mentioned here are set to no access | ||||||
|  | # https://docs.github.com/en/actions/security-guides/automatic-token-authentication#permissions-for-the-github_token | ||||||
|  | permissions: | ||||||
|  |   actions: none | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   release: |   release: | ||||||
|     name: Release |     name: Release | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|  |     permissions: | ||||||
|  |       contents: write  # Required to upload release assets | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v2 |         uses: actions/checkout@v2 | ||||||
| @@ -47,6 +55,13 @@ jobs: | |||||||
|  |  | ||||||
|           script/release |           script/release | ||||||
|  |  | ||||||
|  |       - name: Upload release assets | ||||||
|  |         uses: softprops/action-gh-release@v0.1.14 | ||||||
|  |         with: | ||||||
|  |           files: | | ||||||
|  |             dist/*.whl | ||||||
|  |             dist/*.tar.gz | ||||||
|  |  | ||||||
|   wheels-init: |   wheels-init: | ||||||
|     name: Init wheels build |     name: Init wheels build | ||||||
|     needs: release |     needs: release | ||||||
|   | |||||||
							
								
								
									
										631
									
								
								.yarn/releases/yarn-3.0.2.cjs
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										631
									
								
								.yarn/releases/yarn-3.0.2.cjs
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										785
									
								
								.yarn/releases/yarn-3.2.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										785
									
								
								.yarn/releases/yarn-3.2.0.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -6,4 +6,4 @@ plugins: | |||||||
|   - path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs |   - path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs | ||||||
|     spec: "@yarnpkg/plugin-interactive-tools" |     spec: "@yarnpkg/plugin-interactive-tools" | ||||||
|  |  | ||||||
| yarnPath: .yarn/releases/yarn-3.0.2.cjs | yarnPath: .yarn/releases/yarn-3.2.0.cjs | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ | |||||||
|  |  | ||||||
| This is the repository for the official [Home Assistant](https://home-assistant.io) frontend. | This is the repository for the official [Home Assistant](https://home-assistant.io) frontend. | ||||||
|  |  | ||||||
| [](https://demo.home-assistant.io/) | [](https://demo.home-assistant.io/) | ||||||
|  |  | ||||||
| - [View demo of Home Assistant](https://demo.home-assistant.io/) | - [View demo of Home Assistant](https://demo.home-assistant.io/) | ||||||
| - [More information about Home Assistant](https://home-assistant.io) | - [More information about Home Assistant](https://home-assistant.io) | ||||||
|   | |||||||
| @@ -26,8 +26,8 @@ module.exports = { | |||||||
|   }, |   }, | ||||||
|   version() { |   version() { | ||||||
|     const version = fs |     const version = fs | ||||||
|       .readFileSync(path.resolve(paths.polymer_dir, "setup.cfg"), "utf8") |       .readFileSync(path.resolve(paths.polymer_dir, "pyproject.toml"), "utf8") | ||||||
|       .match(/version\W+=\W(\d{8}\.\d)/); |       .match(/version\W+=\W"(\d{8}\.\d)"/); | ||||||
|     if (!version) { |     if (!version) { | ||||||
|       throw Error("Version not found"); |       throw Error("Version not found"); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ | |||||||
| const gulp = require("gulp"); | const gulp = require("gulp"); | ||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const path = require("path"); | const path = require("path"); | ||||||
| const marked = require("marked"); | const { marked } = require("marked"); | ||||||
| const glob = require("glob"); | const glob = require("glob"); | ||||||
| const yaml = require("js-yaml"); | const yaml = require("js-yaml"); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ const source = require("vinyl-source-stream"); | |||||||
| const vinylBuffer = require("vinyl-buffer"); | const vinylBuffer = require("vinyl-buffer"); | ||||||
| const gulp = require("gulp"); | const gulp = require("gulp"); | ||||||
| const fs = require("fs"); | const fs = require("fs"); | ||||||
| const foreach = require("gulp-foreach"); | const flatmap = require("gulp-flatmap"); | ||||||
| const merge = require("gulp-merge-json"); | const merge = require("gulp-merge-json"); | ||||||
| const rename = require("gulp-rename"); | const rename = require("gulp-rename"); | ||||||
| const transform = require("gulp-json-transform"); | const transform = require("gulp-json-transform"); | ||||||
| @@ -183,7 +183,7 @@ gulp.task("build-merged-translations", () => | |||||||
|     }) |     }) | ||||||
|     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) |     .pipe(transform((data, file) => lokaliseTransform(data, data, file))) | ||||||
|     .pipe( |     .pipe( | ||||||
|       foreach((stream, file) => { |       flatmap((stream, file) => { | ||||||
|         // For each language generate a merged json file. It begins with the master |         // For each language generate a merged json file. It begins with the master | ||||||
|         // translation as a failsafe for untranslated strings, and merges all parent |         // translation as a failsafe for untranslated strings, and merges all parent | ||||||
|         // tags into one file for each specific subtag |         // tags into one file for each specific subtag | ||||||
|   | |||||||
| @@ -3,10 +3,10 @@ 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 { WebpackManifestPlugin } = require("webpack-manifest-plugin"); | const { WebpackManifestPlugin } = require("webpack-manifest-plugin"); | ||||||
| const paths = require("./paths.js"); |  | ||||||
| const bundle = require("./bundle.js"); |  | ||||||
| const log = require("fancy-log"); | const log = require("fancy-log"); | ||||||
| const WebpackBar = require("webpackbar"); | const WebpackBar = require("webpackbar"); | ||||||
|  | const paths = require("./paths.js"); | ||||||
|  | const bundle = require("./bundle.js"); | ||||||
|  |  | ||||||
| class LogStartCompilePlugin { | class LogStartCompilePlugin { | ||||||
|   ignoredFirst = false; |   ignoredFirst = false; | ||||||
| @@ -138,6 +138,8 @@ const createWebpackConfig = ({ | |||||||
|         "lit/directives/cache$": "lit/directives/cache.js", |         "lit/directives/cache$": "lit/directives/cache.js", | ||||||
|         "lit/directives/repeat$": "lit/directives/repeat.js", |         "lit/directives/repeat$": "lit/directives/repeat.js", | ||||||
|         "lit/polyfill-support$": "lit/polyfill-support.js", |         "lit/polyfill-support$": "lit/polyfill-support.js", | ||||||
|  |         "@lit-labs/virtualizer/layouts/grid": | ||||||
|  |           "@lit-labs/virtualizer/layouts/grid.js", | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     output: { |     output: { | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property, query } from "lit/decorators"; | ||||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||||
| 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"; | ||||||
| @@ -20,6 +20,8 @@ class HcLovelace extends LitElement { | |||||||
|  |  | ||||||
|   @property() public urlPath: string | null = null; |   @property() public urlPath: string | null = null; | ||||||
|  |  | ||||||
|  |   @query("hui-view") private _huiView?: HTMLElement; | ||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     const index = this._viewIndex; |     const index = this._viewIndex; | ||||||
|     if (index === undefined) { |     if (index === undefined) { | ||||||
| @@ -78,12 +80,12 @@ class HcLovelace extends LitElement { | |||||||
|           this.lovelaceConfig.background; |           this.lovelaceConfig.background; | ||||||
|  |  | ||||||
|         if (configBackground) { |         if (configBackground) { | ||||||
|           (this.shadowRoot!.querySelector( |           this._huiView!.style.setProperty( | ||||||
|             "hui-view" |  | ||||||
|           ) as HTMLElement)!.style.setProperty( |  | ||||||
|             "--lovelace-background", |             "--lovelace-background", | ||||||
|             configBackground |             configBackground | ||||||
|           ); |           ); | ||||||
|  |         } else { | ||||||
|  |           this._huiView!.style.removeProperty("--lovelace-background"); | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| @@ -116,6 +118,9 @@ class HcLovelace extends LitElement { | |||||||
|       :host > * { |       :host > * { | ||||||
|         flex: 1; |         flex: 1; | ||||||
|       } |       } | ||||||
|  |       hui-view { | ||||||
|  |         background: var(--lovelace-background, var(--primary-background-color)); | ||||||
|  |       } | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import "web-animations-js/web-animations-next-lite.min"; |  | ||||||
| import "../../../src/resources/ha-style"; | import "../../../src/resources/ha-style"; | ||||||
| import "../../../src/resources/roboto"; | import "../../../src/resources/roboto"; | ||||||
| import "./layout/hc-lovelace"; | import "./layout/hc-lovelace"; | ||||||
|   | |||||||
| @@ -194,7 +194,7 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({ | |||||||
|                     type: "state-icon", |                     type: "state-icon", | ||||||
|                     tap_action: { |                     tap_action: { | ||||||
|                       action: "call-service", |                       action: "call-service", | ||||||
|                       service_data: { |                       data: { | ||||||
|                         entity_id: "group.downstairs_lights", |                         entity_id: "group.downstairs_lights", | ||||||
|                       }, |                       }, | ||||||
|                       service: "homeassistant.toggle", |                       service: "homeassistant.toggle", | ||||||
|   | |||||||
| @@ -377,7 +377,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|               name: "AC bed", |               name: "AC bed", | ||||||
|               tap_action: { |               tap_action: { | ||||||
|                 action: "call-service", |                 action: "call-service", | ||||||
|                 service_data: { |                 data: { | ||||||
|                   entity_id: "script.air_cleaner_quiet", |                   entity_id: "script.air_cleaner_quiet", | ||||||
|                 }, |                 }, | ||||||
|                 service: "script.turn_on", |                 service: "script.turn_on", | ||||||
| @@ -390,7 +390,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|               name: "AC bed", |               name: "AC bed", | ||||||
|               tap_action: { |               tap_action: { | ||||||
|                 action: "call-service", |                 action: "call-service", | ||||||
|                 service_data: { |                 data: { | ||||||
|                   entity_id: "script.air_cleaner_auto", |                   entity_id: "script.air_cleaner_auto", | ||||||
|                 }, |                 }, | ||||||
|                 service: "script.turn_on", |                 service: "script.turn_on", | ||||||
| @@ -403,7 +403,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|               name: "AC bed", |               name: "AC bed", | ||||||
|               tap_action: { |               tap_action: { | ||||||
|                 action: "call-service", |                 action: "call-service", | ||||||
|                 service_data: { |                 data: { | ||||||
|                   entity_id: "script.air_cleaner_turbo", |                   entity_id: "script.air_cleaner_turbo", | ||||||
|                 }, |                 }, | ||||||
|                 service: "script.turn_on", |                 service: "script.turn_on", | ||||||
| @@ -416,7 +416,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|               name: "AC", |               name: "AC", | ||||||
|               tap_action: { |               tap_action: { | ||||||
|                 action: "call-service", |                 action: "call-service", | ||||||
|                 service_data: { |                 data: { | ||||||
|                   entity_id: "script.ac_off", |                   entity_id: "script.ac_off", | ||||||
|                 }, |                 }, | ||||||
|                 service: "script.turn_on", |                 service: "script.turn_on", | ||||||
| @@ -429,7 +429,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|               name: "AC", |               name: "AC", | ||||||
|               tap_action: { |               tap_action: { | ||||||
|                 action: "call-service", |                 action: "call-service", | ||||||
|                 service_data: { |                 data: { | ||||||
|                   entity_id: "script.ac_on", |                   entity_id: "script.ac_on", | ||||||
|                 }, |                 }, | ||||||
|                 service: "script.turn_on", |                 service: "script.turn_on", | ||||||
| @@ -629,7 +629,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|                   entity: "scene.morning_lights", |                   entity: "scene.morning_lights", | ||||||
|                   tap_action: { |                   tap_action: { | ||||||
|                     action: "call-service", |                     action: "call-service", | ||||||
|                     service_data: { |                     data: { | ||||||
|                       entity_id: "scene.morning_lights", |                       entity_id: "scene.morning_lights", | ||||||
|                     }, |                     }, | ||||||
|                     service: "scene.turn_on", |                     service: "scene.turn_on", | ||||||
| @@ -641,7 +641,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|                   entity: "scene.movie_time", |                   entity: "scene.movie_time", | ||||||
|                   tap_action: { |                   tap_action: { | ||||||
|                     action: "call-service", |                     action: "call-service", | ||||||
|                     service_data: { |                     data: { | ||||||
|                       entity_id: "scene.movie_time", |                       entity_id: "scene.movie_time", | ||||||
|                     }, |                     }, | ||||||
|                     service: "scene.turn_on", |                     service: "scene.turn_on", | ||||||
| @@ -702,7 +702,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|                   entity: "light.downstairs_lights", |                   entity: "light.downstairs_lights", | ||||||
|                   tap_action: { |                   tap_action: { | ||||||
|                     action: "call-service", |                     action: "call-service", | ||||||
|                     service_data: { |                     data: { | ||||||
|                       entity_id: "light.downstairs_lights", |                       entity_id: "light.downstairs_lights", | ||||||
|                     }, |                     }, | ||||||
|                     service: "light.toggle", |                     service: "light.toggle", | ||||||
| @@ -714,7 +714,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({ | |||||||
|                   entity: "light.upstairs_lights", |                   entity: "light.upstairs_lights", | ||||||
|                   tap_action: { |                   tap_action: { | ||||||
|                     action: "call-service", |                     action: "call-service", | ||||||
|                     service_data: { |                     data: { | ||||||
|                       entity_id: "light.upstairs_lights", |                       entity_id: "light.upstairs_lights", | ||||||
|                     }, |                     }, | ||||||
|                     service: "light.toggle", |                     service: "light.toggle", | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| import { format, startOfToday, startOfTomorrow } from "date-fns"; | import { format, startOfToday, startOfTomorrow } from "date-fns/esm"; | ||||||
| import { EnergySolarForecasts } from "../../../src/data/energy"; | import { EnergySolarForecasts } from "../../../src/data/energy"; | ||||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ export const mockHassioSupervisor = (hass: MockHomeAssistant) => { | |||||||
|             version_latest: "3.6.2", |             version_latest: "3.6.2", | ||||||
|             update_available: false, |             update_available: false, | ||||||
|             repository: "a0d7b954", |             repository: "a0d7b954", | ||||||
|             icon: true, |             icon: false, | ||||||
|             logo: true, |             logo: true, | ||||||
|           }, |           }, | ||||||
|           { |           { | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ import { | |||||||
|   addMonths, |   addMonths, | ||||||
|   differenceInHours, |   differenceInHours, | ||||||
|   endOfDay, |   endOfDay, | ||||||
| } from "date-fns"; | } from "date-fns/esm"; | ||||||
| import { HassEntity } from "home-assistant-js-websocket"; | import { HassEntity } from "home-assistant-js-websocket"; | ||||||
| import { StatisticValue } from "../../../src/data/history"; | import { StatisticValue } from "../../../src/data/history"; | ||||||
| import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | import { MockHomeAssistant } from "../../../src/fake_data/provide_hass"; | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								gallery/public/api/hassio/addons/core_zwave_js/icon
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/api/hassio/addons/core_zwave_js/icon
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/clearspace.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/clearspace.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 44 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo-variants.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo-variants.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 35 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo-with-text.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo-with-text.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 67 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 27 KiB | 
							
								
								
									
										
											BIN
										
									
								
								gallery/public/images/using-our-logo.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gallery/public/images/using-our-logo.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 32 KiB | 
| @@ -23,7 +23,7 @@ if [[ "${PULL_REQUEST}" == "true" ]]; then | |||||||
|     createStatus "pending" "Building design preview" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" |     createStatus "pending" "Building design preview" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||||
|     gulp build-gallery |     gulp build-gallery | ||||||
|     if [ $? -eq 0 ]; then |     if [ $? -eq 0 ]; then | ||||||
|       createStatus "success" "Build complete" "$DEPLOY_URL" |       createStatus "success" "Build complete" "$DEPLOY_PRIME_URL" | ||||||
|     else |     else | ||||||
|       createStatus "error" "Build failed" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" |       createStatus "error" "Build failed" "https://app.netlify.com/sites/home-assistant-gallery/deploys/$BUILD_ID" | ||||||
|     fi |     fi | ||||||
|   | |||||||
| @@ -36,12 +36,17 @@ module.exports = [ | |||||||
|     category: "misc", |     category: "misc", | ||||||
|     header: "Miscelaneous", |     header: "Miscelaneous", | ||||||
|   }, |   }, | ||||||
|  |   { | ||||||
|  |     category: "brand", | ||||||
|  |     header: "Brand", | ||||||
|  |   }, | ||||||
|   { |   { | ||||||
|     category: "user-test", |     category: "user-test", | ||||||
|     header: "User Tests", |     header: "Users", | ||||||
|  |     pages: ["user-types", "configuration-menu"], | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     category: "design.home-assistant.io", |     category: "design.home-assistant.io", | ||||||
|     header: "Design Documentation", |     header: "About", | ||||||
|   }, |   }, | ||||||
| ]; | ]; | ||||||
|   | |||||||
| @@ -53,13 +53,19 @@ class DemoBlackWhiteRow extends LitElement { | |||||||
|  |  | ||||||
|   firstUpdated(changedProps) { |   firstUpdated(changedProps) { | ||||||
|     super.firstUpdated(changedProps); |     super.firstUpdated(changedProps); | ||||||
|     applyThemesOnElement(this.shadowRoot!.querySelector(".dark"), { |     applyThemesOnElement( | ||||||
|  |       this.shadowRoot!.querySelector(".dark"), | ||||||
|  |       { | ||||||
|         default_theme: "default", |         default_theme: "default", | ||||||
|         default_dark_theme: "default", |         default_dark_theme: "default", | ||||||
|         themes: {}, |         themes: {}, | ||||||
|         darkMode: true, |         darkMode: true, | ||||||
|         theme: "default", |         theme: "default", | ||||||
|     }); |       }, | ||||||
|  |       undefined, | ||||||
|  |       undefined, | ||||||
|  |       true | ||||||
|  |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleSubmit(ev) { |   handleSubmit(ev) { | ||||||
|   | |||||||
| @@ -78,6 +78,9 @@ class DemoCards extends LitElement { | |||||||
|     ha-formfield { |     ha-formfield { | ||||||
|       margin-right: 16px; |       margin-right: 16px; | ||||||
|     } |     } | ||||||
|  |     #container { | ||||||
|  |       background-color: var(--primary-background-color); | ||||||
|  |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -12,7 +12,14 @@ class PageDescription extends HaMarkdown { | |||||||
|     if (!PAGES[this.page].description) { |     if (!PAGES[this.page].description) { | ||||||
|       return html``; |       return html``; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|  |       <div class="heading"> | ||||||
|  |         <div class="title"> | ||||||
|  |           ${PAGES[this.page].metadata.title || this.page.split("/")[1]} | ||||||
|  |         </div> | ||||||
|  |         <div class="subtitle">${PAGES[this.page].metadata.subtitle}</div> | ||||||
|  |       </div> | ||||||
|       ${until( |       ${until( | ||||||
|         PAGES[this.page] |         PAGES[this.page] | ||||||
|           .description() |           .description() | ||||||
| @@ -25,9 +32,22 @@ class PageDescription extends HaMarkdown { | |||||||
|   static styles = [ |   static styles = [ | ||||||
|     HaMarkdown.styles, |     HaMarkdown.styles, | ||||||
|     css` |     css` | ||||||
|  |       .heading { | ||||||
|  |         padding: 16px; | ||||||
|  |         border-bottom: 1px solid var(--secondary-background-color); | ||||||
|  |       } | ||||||
|  |       .title { | ||||||
|  |         font-size: 42px; | ||||||
|  |         line-height: 56px; | ||||||
|  |         padding-bottom: 8px; | ||||||
|  |       } | ||||||
|  |       .subtitle { | ||||||
|  |         font-size: 18px; | ||||||
|  |         line-height: 24px; | ||||||
|  |       } | ||||||
|       .root { |       .root { | ||||||
|         max-width: 800px; |         max-width: 800px; | ||||||
|         margin: 0 auto; |         margin: 16px auto; | ||||||
|       } |       } | ||||||
|       .root > *:first-child { |       .root > *:first-child { | ||||||
|         margin-top: 0; |         margin-top: 0; | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								gallery/src/data/text.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								gallery/src/data/text.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | export const LONG_TEXT = ` | ||||||
|  | Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nunc laoreet velit ut elit volutpat, eget ultrices odio lacinia. In imperdiet malesuada est, nec sagittis metus ultricies quis. Sed nisl ex, convallis porttitor ante quis, hendrerit tristique justo. Mauris pharetra venenatis augue, eu maximus sem cursus in. Quisque sed consequat risus. Suspendisse facilisis ligula a odio consectetur condimentum. Curabitur vehicula elit nec augue mollis, et volutpat massa dictum. | ||||||
|  |  | ||||||
|  | Nam pellentesque auctor rutrum. Suspendisse elit est, sodales vel diam nec, porttitor faucibus massa. Ut pretium ac orci eu pharetra. Praesent in nibh at magna viverra rutrum eu vitae tortor. Etiam eget sem ex. Fusce tristique odio nec lacus mattis, vitae tempor nunc malesuada. Maecenas faucibus magna vel libero maximus egestas. Vestibulum luctus semper velit, in lobortis risus tempus non. Curabitur bibendum ornare commodo. Quisque commodo neque sit amet tincidunt lacinia. Proin elementum ante velit, eu congue nulla semper quis. Pellentesque consequat vel nunc at scelerisque. Mauris sit amet venenatis diam, blandit viverra leo. Integer commodo laoreet orci. | ||||||
|  |  | ||||||
|  | Curabitur ipsum tortor, sodales ut augue sed, commodo porttitor libero. Pellentesque molestie vitae mi consectetur tempor. In sed lectus consequat, lobortis neque non, semper ipsum. Etiam eget ex et nibh sagittis pulvinar lacinia ac mauris. Aenean ligula eros, viverra ac nibh at, venenatis semper quam. Sed interdum ligula sit amet massa tincidunt tincidunt. Suspendisse potenti. Aliquam egestas facilisis est, sed faucibus erat scelerisque id. Duis dolor quam, viverra vitae orci euismod, laoreet pellentesque justo. Nunc malesuada non erat at ullamcorper. Mauris eget posuere odio. Vestibulum turpis nunc, pharetra eget ante in, feugiat mollis justo. Proin porttitor, diam nec vulputate pretium, tellus arcu rhoncus turpis, a blandit nisi nulla quis arcu. Nunc ac ullamcorper ligula, nec facilisis leo. | ||||||
|  |  | ||||||
|  | In vitae eros sollicitudin, iaculis ex eget, egestas orci. Etiam sed pretium lorem. Nam nisi enim, consectetur sit amet semper ac, semper pharetra diam. In pulvinar neque sapien, ac ullamcorper est lacinia a. Etiam tincidunt velit sed diam malesuada, eu ornare ex consectetur. Phasellus in imperdiet tellus. Sed bibendum, dui sit amet fringilla aliquet, enim odio sollicitudin lorem, vel semper turpis mauris vel mauris. Aenean congue magna ac massa cursus, in dictum orci commodo. Pellentesque mollis velit in sollicitudin tincidunt. Vestibulum et efficitur nulla. | ||||||
|  |  | ||||||
|  | Quisque posuere, velit sed porttitor dapibus, neque augue fringilla felis, eu luctus nisi nisl nec ipsum. Curabitur pellentesque ac lectus eget ultricies. Vestibulum est dolor, lacinia pharetra vulputate a, facilisis a magna. Nam vitae arcu nibh. Praesent finibus blandit ante, ac gravida ex mollis eget. Donec quam est, pulvinar vitae neque ut, bibendum aliquam erat. Nullam mollis arcu at sem tincidunt, in tristique lectus facilisis. Aenean ut lacus vel nisl finibus iaculis non a turpis. Integer eget ipsum ante. Donec nunc neque, vestibulum ac magna ac, posuere scelerisque dui. Pellentesque massa nibh, rhoncus id dolor quis, placerat posuere turpis. Donec aliquet augue nisi, eu finibus dui auctor et. Vestibulum eu varius lorem. Quisque lectus ante, malesuada pretium risus eget, interdum mattis enim. | ||||||
|  | `; | ||||||
| @@ -119,7 +119,7 @@ export const basicTrace: DemoTrace = { | |||||||
|             params: { |             params: { | ||||||
|               domain: "input_boolean", |               domain: "input_boolean", | ||||||
|               service: "toggle", |               service: "toggle", | ||||||
|               service_data: {}, |               data: {}, | ||||||
|               target: { |               target: { | ||||||
|                 entity_id: ["input_boolean.toggle_4"], |                 entity_id: ["input_boolean.toggle_4"], | ||||||
|               }, |               }, | ||||||
| @@ -164,7 +164,7 @@ export const basicTrace: DemoTrace = { | |||||||
|             params: { |             params: { | ||||||
|               domain: "input_boolean", |               domain: "input_boolean", | ||||||
|               service: "toggle", |               service: "toggle", | ||||||
|               service_data: {}, |               data: {}, | ||||||
|               target: { |               target: { | ||||||
|                 entity_id: ["input_boolean.toggle_2"], |                 entity_id: ["input_boolean.toggle_2"], | ||||||
|               }, |               }, | ||||||
| @@ -182,7 +182,7 @@ export const basicTrace: DemoTrace = { | |||||||
|             params: { |             params: { | ||||||
|               domain: "input_boolean", |               domain: "input_boolean", | ||||||
|               service: "toggle", |               service: "toggle", | ||||||
|               service_data: {}, |               data: {}, | ||||||
|               target: { |               target: { | ||||||
|                 entity_id: ["input_boolean.toggle_3"], |                 entity_id: ["input_boolean.toggle_3"], | ||||||
|               }, |               }, | ||||||
| @@ -200,7 +200,7 @@ export const basicTrace: DemoTrace = { | |||||||
|             params: { |             params: { | ||||||
|               domain: "input_boolean", |               domain: "input_boolean", | ||||||
|               service: "toggle", |               service: "toggle", | ||||||
|               service_data: {}, |               data: {}, | ||||||
|               target: { |               target: { | ||||||
|                 entity_id: ["input_boolean.toggle_4"], |                 entity_id: ["input_boolean.toggle_4"], | ||||||
|               }, |               }, | ||||||
| @@ -298,11 +298,11 @@ export const basicTrace: DemoTrace = { | |||||||
|       source: "state of input_boolean.toggle_1", |       source: "state of input_boolean.toggle_1", | ||||||
|       entity_id: "automation.toggle_toggles", |       entity_id: "automation.toggle_toggles", | ||||||
|       context_id: "6cfcae368e7b3686fad6c59e83ae76c9", |       context_id: "6cfcae368e7b3686fad6c59e83ae76c9", | ||||||
|       when: "2021-03-25T04:36:51.240832+00:00", |       when: 1616647011.240832, | ||||||
|       domain: "automation", |       domain: "automation", | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       when: "2021-03-25T04:36:51.249828+00:00", |       when: 1616647011.249828, | ||||||
|       name: "Toggle 4", |       name: "Toggle 4", | ||||||
|       state: "on", |       state: "on", | ||||||
|       entity_id: "input_boolean.toggle_4", |       entity_id: "input_boolean.toggle_4", | ||||||
| @@ -313,7 +313,7 @@ export const basicTrace: DemoTrace = { | |||||||
|       context_name: "Ensure Party mode", |       context_name: "Ensure Party mode", | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       when: "2021-03-25T04:36:51.258947+00:00", |       when: 1616647011.258947, | ||||||
|       name: "Toggle 2", |       name: "Toggle 2", | ||||||
|       state: "on", |       state: "on", | ||||||
|       entity_id: "input_boolean.toggle_2", |       entity_id: "input_boolean.toggle_2", | ||||||
| @@ -324,7 +324,7 @@ export const basicTrace: DemoTrace = { | |||||||
|       context_name: "Ensure Party mode", |       context_name: "Ensure Party mode", | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       when: "2021-03-25T04:36:51.261806+00:00", |       when: 1616647011.261806, | ||||||
|       name: "Toggle 3", |       name: "Toggle 3", | ||||||
|       state: "off", |       state: "off", | ||||||
|       entity_id: "input_boolean.toggle_3", |       entity_id: "input_boolean.toggle_3", | ||||||
| @@ -335,7 +335,7 @@ export const basicTrace: DemoTrace = { | |||||||
|       context_name: "Ensure Party mode", |       context_name: "Ensure Party mode", | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       when: "2021-03-25T04:36:51.265246+00:00", |       when: 1616647011.265246, | ||||||
|       name: "Toggle 4", |       name: "Toggle 4", | ||||||
|       state: "off", |       state: "off", | ||||||
|       entity_id: "input_boolean.toggle_4", |       entity_id: "input_boolean.toggle_4", | ||||||
|   | |||||||
| @@ -185,11 +185,11 @@ export const motionLightTrace: DemoTrace = { | |||||||
|         "has been triggered by state of binary_sensor.pauluss_macbook_pro_camera_in_use", |         "has been triggered by state of binary_sensor.pauluss_macbook_pro_camera_in_use", | ||||||
|       source: "state of binary_sensor.pauluss_macbook_pro_camera_in_use", |       source: "state of binary_sensor.pauluss_macbook_pro_camera_in_use", | ||||||
|       entity_id: "automation.auto_elgato", |       entity_id: "automation.auto_elgato", | ||||||
|       when: "2021-03-14T06:07:01.768492+00:00", |       when: 1615702021.768492, | ||||||
|       domain: "automation", |       domain: "automation", | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       when: "2021-03-14T06:07:01.872187+00:00", |       when: 1615702021.872187, | ||||||
|       name: "Elgato Key Light Air", |       name: "Elgato Key Light Air", | ||||||
|       state: "on", |       state: "on", | ||||||
|       entity_id: "light.elgato_key_light_air", |       entity_id: "light.elgato_key_light_air", | ||||||
| @@ -200,7 +200,7 @@ export const motionLightTrace: DemoTrace = { | |||||||
|       context_name: "Auto Elgato", |       context_name: "Auto Elgato", | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       when: "2021-03-14T06:07:53.284505+00:00", |       when: 1615702073.284505, | ||||||
|       name: "Elgato Key Light Air", |       name: "Elgato Key Light Air", | ||||||
|       state: "off", |       state: "off", | ||||||
|       entity_id: "light.elgato_key_light_air", |       entity_id: "light.elgato_key_light_air", | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import { html, css, LitElement, PropertyValues } from "lit"; | |||||||
| import { customElement, property, query } from "lit/decorators"; | import { customElement, property, query } from "lit/decorators"; | ||||||
| import "../../src/components/ha-icon-button"; | import "../../src/components/ha-icon-button"; | ||||||
| import "../../src/managers/notification-manager"; | import "../../src/managers/notification-manager"; | ||||||
|  | import "../../src/components/ha-expansion-panel"; | ||||||
| import { haStyle } from "../../src/resources/styles"; | import { haStyle } from "../../src/resources/styles"; | ||||||
| import { PAGES, SIDEBAR } from "../build/import-pages"; | import { PAGES, SIDEBAR } from "../build/import-pages"; | ||||||
| import { dynamicElement } from "../../src/common/dom/dynamic-element-directive"; | import { dynamicElement } from "../../src/common/dom/dynamic-element-directive"; | ||||||
| @@ -44,6 +45,10 @@ class HaGallery extends LitElement { | |||||||
|       for (const page of group.pages!) { |       for (const page of group.pages!) { | ||||||
|         const key = `${group.category}/${page}`; |         const key = `${group.category}/${page}`; | ||||||
|         const active = this._page === key; |         const active = this._page === key; | ||||||
|  |         if (!(key in PAGES)) { | ||||||
|  |           console.error("Undefined page referenced in sidebar.js:", key); | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|         const title = PAGES[key].metadata.title || page; |         const title = PAGES[key].metadata.title || page; | ||||||
|         links.push(html` |         links.push(html` | ||||||
|           <a ?active=${active} href=${`#${group.category}/${page}`}>${title}</a> |           <a ?active=${active} href=${`#${group.category}/${page}`}>${title}</a> | ||||||
| @@ -53,10 +58,9 @@ class HaGallery extends LitElement { | |||||||
|       sidebar.push( |       sidebar.push( | ||||||
|         group.header |         group.header | ||||||
|           ? html` |           ? html` | ||||||
|               <details> |               <ha-expansion-panel .header=${group.header}> | ||||||
|                 <summary class="section">${group.header}</summary> |  | ||||||
|                 ${links} |                 ${links} | ||||||
|               </details> |               </ha-expansion-panel> | ||||||
|             ` |             ` | ||||||
|           : links |           : links | ||||||
|       ); |       ); | ||||||
| @@ -92,6 +96,12 @@ class HaGallery extends LitElement { | |||||||
|             ${dynamicElement(`demo-${this._page.replace("/", "-")}`)} |             ${dynamicElement(`demo-${this._page.replace("/", "-")}`)} | ||||||
|           </div> |           </div> | ||||||
|           <div class="page-footer"> |           <div class="page-footer"> | ||||||
|  |             <div class="header">Help us to improve our documentation</div> | ||||||
|  |             <div class="secondary"> | ||||||
|  |               Suggest an edit to this page, or provide/view feedback for this | ||||||
|  |               page. | ||||||
|  |             </div> | ||||||
|  |             <div> | ||||||
|               ${PAGES[this._page].description || |               ${PAGES[this._page].description || | ||||||
|               Object.keys(PAGES[this._page].metadata).length > 0 |               Object.keys(PAGES[this._page].metadata).length > 0 | ||||||
|                 ? html` |                 ? html` | ||||||
| @@ -115,6 +125,7 @@ class HaGallery extends LitElement { | |||||||
|                 : ""} |                 : ""} | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|  |         </div> | ||||||
|       </mwc-drawer> |       </mwc-drawer> | ||||||
|       <notification-manager |       <notification-manager | ||||||
|         .hass=${FAKE_HASS} |         .hass=${FAKE_HASS} | ||||||
| @@ -186,27 +197,16 @@ class HaGallery extends LitElement { | |||||||
|         padding: 4px; |         padding: 4px; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .sidebar details { |  | ||||||
|         margin-top: 1em; |  | ||||||
|         margin-left: 1em; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       .sidebar summary { |  | ||||||
|         cursor: pointer; |  | ||||||
|         font-weight: bold; |  | ||||||
|         margin-bottom: 8px; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       .sidebar a { |       .sidebar a { | ||||||
|         color: var(--primary-text-color); |         color: var(--primary-text-color); | ||||||
|         display: block; |         display: block; | ||||||
|         padding: 4px 12px; |         padding: 12px; | ||||||
|         text-decoration: none; |         text-decoration: none; | ||||||
|         position: relative; |         position: relative; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .sidebar a[active]::before { |       .sidebar a[active]::before { | ||||||
|         border-radius: 4px; |         border-radius: 12px; | ||||||
|         position: absolute; |         position: absolute; | ||||||
|         top: 0; |         top: 0; | ||||||
|         right: 2px; |         right: 2px; | ||||||
| @@ -237,14 +237,32 @@ class HaGallery extends LitElement { | |||||||
|  |  | ||||||
|       .page-footer { |       .page-footer { | ||||||
|         text-align: center; |         text-align: center; | ||||||
|         margin: 16px 0; |         margin: 16px; | ||||||
|         padding-top: 16px; |         padding: 16px; | ||||||
|         border-top: 1px solid rgba(0, 0, 0, 0.12); |         border-radius: 12px; | ||||||
|  |         background-color: var(--primary-background-color); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       .page-footer div { | ||||||
|  |         margin-top: 4px; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       .page-footer .header { | ||||||
|  |         font-size: 16px; | ||||||
|  |         font-weight: 500; | ||||||
|  |         line-height: 28px; | ||||||
|  |         text-align: center; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       .page-footer .secondary { | ||||||
|  |         line-height: 23px; | ||||||
|  |         text-align: center; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .page-footer a { |       .page-footer a { | ||||||
|         display: inline-block; |         display: inline-block; | ||||||
|         margin: 0 8px; |         margin: 0 8px; | ||||||
|  |         text-decoration: none; | ||||||
|       } |       } | ||||||
|     `, |     `, | ||||||
|   ]; |   ]; | ||||||
|   | |||||||
| @@ -62,6 +62,45 @@ const ACTIONS = [ | |||||||
|       entity_id: "input_boolean.toggle_4", |       entity_id: "input_boolean.toggle_4", | ||||||
|     }, |     }, | ||||||
|   }, |   }, | ||||||
|  |   { | ||||||
|  |     parallel: [ | ||||||
|  |       { scene: "scene.kitchen_morning" }, | ||||||
|  |       { | ||||||
|  |         service: "media_player.play_media", | ||||||
|  |         target: { entity_id: "media_player.living_room" }, | ||||||
|  |         data: { media_content_id: "", media_content_type: "" }, | ||||||
|  |         metadata: { title: "Happy Song" }, | ||||||
|  |       }, | ||||||
|  |     ], | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     stop: "No one is home!", | ||||||
|  |   }, | ||||||
|  |   { repeat: { count: 3, sequence: [{ delay: "00:00:01" }] } }, | ||||||
|  |   { | ||||||
|  |     repeat: { | ||||||
|  |       for_each: ["bread", "butter", "cheese"], | ||||||
|  |       sequence: [{ delay: "00:00:01" }], | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     if: [{ condition: "state" }], | ||||||
|  |     then: [{ delay: "00:00:01" }], | ||||||
|  |     else: [{ delay: "00:00:05" }], | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     choose: [ | ||||||
|  |       { | ||||||
|  |         conditions: [{ condition: "state" }], | ||||||
|  |         sequence: [{ delay: "00:00:01" }], | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         conditions: [{ condition: "sun" }], | ||||||
|  |         sequence: [{ delay: "00:00:05" }], | ||||||
|  |       }, | ||||||
|  |     ], | ||||||
|  |     default: [{ delay: "00:00:03" }], | ||||||
|  |   }, | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| @customElement("demo-automation-describe-action") | @customElement("demo-automation-describe-action") | ||||||
|   | |||||||
| @@ -20,6 +20,10 @@ import { HaWaitForTriggerAction } from "../../../../src/panels/config/automation | |||||||
| import { HaWaitAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-wait_template"; | import { HaWaitAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-wait_template"; | ||||||
| import { Action } from "../../../../src/data/script"; | import { Action } from "../../../../src/data/script"; | ||||||
| import { HaConditionAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-condition"; | import { HaConditionAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-condition"; | ||||||
|  | import { HaParallelAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-parallel"; | ||||||
|  | import { HaIfAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-if"; | ||||||
|  | import { HaStopAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-stop"; | ||||||
|  | import { HaPlayMediaAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-play_media"; | ||||||
|  |  | ||||||
| const SCHEMAS: { name: string; actions: Action[] }[] = [ | const SCHEMAS: { name: string; actions: Action[] }[] = [ | ||||||
|   { name: "Event", actions: [HaEventAction.defaultConfig] }, |   { name: "Event", actions: [HaEventAction.defaultConfig] }, | ||||||
| @@ -28,11 +32,15 @@ const SCHEMAS: { name: string; actions: Action[] }[] = [ | |||||||
|   { name: "Condition", actions: [HaConditionAction.defaultConfig] }, |   { name: "Condition", actions: [HaConditionAction.defaultConfig] }, | ||||||
|   { name: "Delay", actions: [HaDelayAction.defaultConfig] }, |   { name: "Delay", actions: [HaDelayAction.defaultConfig] }, | ||||||
|   { name: "Scene", actions: [HaSceneAction.defaultConfig] }, |   { name: "Scene", actions: [HaSceneAction.defaultConfig] }, | ||||||
|  |   { name: "Play media", actions: [HaPlayMediaAction.defaultConfig] }, | ||||||
|   { name: "Wait", actions: [HaWaitAction.defaultConfig] }, |   { name: "Wait", actions: [HaWaitAction.defaultConfig] }, | ||||||
|   { name: "WaitForTrigger", actions: [HaWaitForTriggerAction.defaultConfig] }, |   { name: "WaitForTrigger", actions: [HaWaitForTriggerAction.defaultConfig] }, | ||||||
|   { name: "Repeat", actions: [HaRepeatAction.defaultConfig] }, |   { name: "Repeat", actions: [HaRepeatAction.defaultConfig] }, | ||||||
|  |   { name: "If-Then", actions: [HaIfAction.defaultConfig] }, | ||||||
|   { name: "Choose", actions: [HaChooseAction.defaultConfig] }, |   { name: "Choose", actions: [HaChooseAction.defaultConfig] }, | ||||||
|   { name: "Variables", actions: [{ variables: { hello: "1" } }] }, |   { name: "Variables", actions: [{ variables: { hello: "1" } }] }, | ||||||
|  |   { name: "Parallel", actions: [HaParallelAction.defaultConfig] }, | ||||||
|  |   { name: "Stop", actions: [HaStopAction.defaultConfig] }, | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| @customElement("demo-automation-editor-action") | @customElement("demo-automation-editor-action") | ||||||
| @@ -86,6 +94,6 @@ class DemoHaAutomationEditorAction extends LitElement { | |||||||
|  |  | ||||||
| declare global { | declare global { | ||||||
|   interface HTMLElementTagNameMap { |   interface HTMLElementTagNameMap { | ||||||
|     "demo-ha-automation-editor-action": DemoHaAutomationEditorAction; |     "demo-automation-editor-action": DemoHaAutomationEditorAction; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | |||||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||||
| import type { Condition } from "../../../../src/data/automation"; | import type { ConditionWithShorthand } from "../../../../src/data/automation"; | ||||||
| import "../../../../src/panels/config/automation/condition/ha-automation-condition"; | import "../../../../src/panels/config/automation/condition/ha-automation-condition"; | ||||||
| import { HaDeviceCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-device"; | import { HaDeviceCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-device"; | ||||||
| import { HaLogicalCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-logical"; | import { HaLogicalCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-logical"; | ||||||
| @@ -20,7 +20,7 @@ import { HaTimeCondition } from "../../../../src/panels/config/automation/condit | |||||||
| import { HaTriggerCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-trigger"; | import { HaTriggerCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-trigger"; | ||||||
| import { HaZoneCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-zone"; | import { HaZoneCondition } from "../../../../src/panels/config/automation/condition/types/ha-automation-condition-zone"; | ||||||
|  |  | ||||||
| const SCHEMAS: { name: string; conditions: Condition[] }[] = [ | const SCHEMAS: { name: string; conditions: ConditionWithShorthand[] }[] = [ | ||||||
|   { |   { | ||||||
|     name: "State", |     name: "State", | ||||||
|     conditions: [{ condition: "state", ...HaStateCondition.defaultConfig }], |     conditions: [{ condition: "state", ...HaStateCondition.defaultConfig }], | ||||||
| @@ -69,6 +69,14 @@ const SCHEMAS: { name: string; conditions: Condition[] }[] = [ | |||||||
|     name: "Trigger", |     name: "Trigger", | ||||||
|     conditions: [{ condition: "trigger", ...HaTriggerCondition.defaultConfig }], |     conditions: [{ condition: "trigger", ...HaTriggerCondition.defaultConfig }], | ||||||
|   }, |   }, | ||||||
|  |   { | ||||||
|  |     name: "Shorthand", | ||||||
|  |     conditions: [ | ||||||
|  |       { and: HaLogicalCondition.defaultConfig.conditions }, | ||||||
|  |       { or: HaLogicalCondition.defaultConfig.conditions }, | ||||||
|  |       { not: HaLogicalCondition.defaultConfig.conditions }, | ||||||
|  |     ], | ||||||
|  |   }, | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| @customElement("demo-automation-editor-condition") | @customElement("demo-automation-editor-condition") | ||||||
|   | |||||||
							
								
								
									
										34
									
								
								gallery/src/pages/brand/logo.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								gallery/src/pages/brand/logo.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | |||||||
|  | --- | ||||||
|  | title: "Logo" | ||||||
|  | --- | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # Using our logo | ||||||
|  |  | ||||||
|  | As a community, we are proud of our logo. Follow these guidelines to ensure it always looks its best. Our logo follows Google's material design spec and uses the blue interface color. | ||||||
|  |  | ||||||
|  | [Download Logo](https://github.com/home-assistant/assets/tree/master/logo) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Using the icon | ||||||
|  |  | ||||||
|  | Our icon is a shorter and most used version of our logo. The icon can exist without the wordmark, the wordmark should never exist without the icon. | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Using the right variant | ||||||
|  |  | ||||||
|  | The pretty blue logo with a background shadow, pictured top left, is our primary logo. It should only be used with black, white, and non-duotone photography. | ||||||
|  |  | ||||||
|  | When needed you can use our logo without a shadow, as seen as the second variant.  | ||||||
|  |  | ||||||
|  | The outlined logo should only be used on packaging. | ||||||
|  |  | ||||||
|  | ## Exclusion zone | ||||||
|  |  | ||||||
|  | The logo needs some personal space. It's exclusion zone is equal to a quarter the height of the icon. | ||||||
|  |  | ||||||
|  |  | ||||||
							
								
								
									
										41
									
								
								gallery/src/pages/brand/our-story.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								gallery/src/pages/brand/our-story.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | |||||||
|  | --- | ||||||
|  | title: "Our story" | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | ## Open source home automation that puts local control and privacy first | ||||||
|  |  | ||||||
|  | Home Assistant is a free and open-source software for home automation that is designed to be the central control system for smart home devices with a focus on local control and privacy. It can be accessed via a web-based user interface, via apps for Android and iOS, or using voice commands via a supported virtual assistant like Google Assistant and Amazon Alexa. | ||||||
|  |  | ||||||
|  | IoT devices and services are supported by modular support for controlling proprietary ecosystems if they provide public access via an Open API for third-party integrations and protocols like Bluetooth, MQTT, Zigbee, and Z-Wave, After the Home Assistant software application is installed as a computer appliance it will act as a central control system for home automation. Information from all entities it sees can be used and controlled from within scripts trigger automations using scheduling and "blueprint" subroutines, e.g. for controlling lighting, climate, entertainment systems, and appliances. | ||||||
|  |  | ||||||
|  | # Open Home | ||||||
|  |  | ||||||
|  | The Open Home is our vision for the smart home. It defines the values that we put at the heart of every decision we make at Home Assistant. It’s woven into our architecture, licensing, community, and everything else. | ||||||
|  |  | ||||||
|  | The Open Home is about privacy, choice, and durability. | ||||||
|  |  | ||||||
|  | ## Privacy | ||||||
|  |  | ||||||
|  | Your home should be your safe space. A place where you can be your true self without having to bother about what the world thinks of you. A place where you don’t need to act differently to avoid an algorithm categorizing your behavior. Privacy for the Open Home means that devices need to work locally. No one else needs to know if you turn on a light bulb or change the thermostat. | ||||||
|  |  | ||||||
|  | It is okay for a product to offer a cloud connection, but it should be extra and opt-in. | ||||||
|  |  | ||||||
|  | ## Choice | ||||||
|  |  | ||||||
|  | Devices in your home gather data about themselves and their surroundings. Your data. Vendors shouldn’t be able to limit your access to your data or limit the interoperability of your devices with the rest of your smart home. | ||||||
|  |  | ||||||
|  | Choice for the Open Home means that devices need to make the gathered data available through local APIs. This avoids vendor lock-in and allows users to create their own smart home with devices from different manufacturers. | ||||||
|  |  | ||||||
|  | ## Durability | ||||||
|  |  | ||||||
|  | If there is one thing that technology firms are very good at, it is launching new products. However, maintaining the products and making sure they keep working is an afterthought for most. The result is that vendors can decide to no longer support your device, crippling its features or even preventing it from working at all. As we install more and more devices in our home, durability is becoming more and more important. We shouldn’t have to buy everything new every couple of years because the manufacturer decided to move on. | ||||||
|  |  | ||||||
|  | Durability for the Open Home means that devices are designed and built to keep working. Not just this year, but for the next decade. | ||||||
|  |  | ||||||
|  | # Our history | ||||||
|  |  | ||||||
|  | The project was started as a Python application by Paulus Schoutsen in September 2013 and first published publicly on GitHub in November 2013. In July 2017, a managed operating system called Hass.io was initially introduced to make it easier use to use Home Assistant on single-board computers like the Raspberry Pi series. Its bundled "supervisor" management system allowed users to manage, backup, and update the local installation and introduced the option to extend the functionality of the software with add-ons. | ||||||
|  |  | ||||||
|  | An optional subscription service was introduced in December 2017 for $5/month to solve the complexities associated with secured remote access, as well as linking to Amazon Alexa and Google Assistant. Nabu Casa, Inc. was formed in September 2018 to take over the subscription service. The company's funding is based solely on revenue from the subscription service. It is used to finance the project's infrastructure and to pay for full-time employees contributing to the project. | ||||||
|  |  | ||||||
|  | In January 2020, branding was adjusted to make it easier to refer to different parts of the project. The main piece of software was renamed to Home Assistant Core, while the full suite of software with the embedded operating system and bundled "supervisor" management system was renamed to Home Assistant. | ||||||
| @@ -1,5 +1,6 @@ | |||||||
| --- | --- | ||||||
| title: Alerts | title: Alerts | ||||||
|  | subtitle: An alert displays a short, important message in a way that attracts the user's attention without interrupting the user's task. | ||||||
| --- | --- | ||||||
|  |  | ||||||
| # Alert `<ha-alert>` | # Alert `<ha-alert>` | ||||||
|   | |||||||
| @@ -159,13 +159,19 @@ export class DemoHaAlert extends LitElement { | |||||||
|  |  | ||||||
|   firstUpdated(changedProps) { |   firstUpdated(changedProps) { | ||||||
|     super.firstUpdated(changedProps); |     super.firstUpdated(changedProps); | ||||||
|     applyThemesOnElement(this.shadowRoot!.querySelector(".dark"), { |     applyThemesOnElement( | ||||||
|  |       this.shadowRoot!.querySelector(".dark"), | ||||||
|  |       { | ||||||
|         default_theme: "default", |         default_theme: "default", | ||||||
|         default_dark_theme: "default", |         default_dark_theme: "default", | ||||||
|         themes: {}, |         themes: {}, | ||||||
|         darkMode: true, |         darkMode: true, | ||||||
|         theme: "default", |         theme: "default", | ||||||
|     }); |       }, | ||||||
|  |       undefined, | ||||||
|  |       undefined, | ||||||
|  |       true | ||||||
|  |     ); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static get styles() { |   static get styles() { | ||||||
|   | |||||||
| @@ -3,18 +3,7 @@ import { customElement } from "lit/decorators"; | |||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
| import "../../../../src/components/ha-faded"; | import "../../../../src/components/ha-faded"; | ||||||
| import "../../../../src/components/ha-markdown"; | import "../../../../src/components/ha-markdown"; | ||||||
|  | import { LONG_TEXT } from "../../data/text"; | ||||||
| const LONG_TEXT = ` |  | ||||||
| Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nunc laoreet velit ut elit volutpat, eget ultrices odio lacinia. In imperdiet malesuada est, nec sagittis metus ultricies quis. Sed nisl ex, convallis porttitor ante quis, hendrerit tristique justo. Mauris pharetra venenatis augue, eu maximus sem cursus in. Quisque sed consequat risus. Suspendisse facilisis ligula a odio consectetur condimentum. Curabitur vehicula elit nec augue mollis, et volutpat massa dictum. |  | ||||||
|  |  | ||||||
| Nam pellentesque auctor rutrum. Suspendisse elit est, sodales vel diam nec, porttitor faucibus massa. Ut pretium ac orci eu pharetra. Praesent in nibh at magna viverra rutrum eu vitae tortor. Etiam eget sem ex. Fusce tristique odio nec lacus mattis, vitae tempor nunc malesuada. Maecenas faucibus magna vel libero maximus egestas. Vestibulum luctus semper velit, in lobortis risus tempus non. Curabitur bibendum ornare commodo. Quisque commodo neque sit amet tincidunt lacinia. Proin elementum ante velit, eu congue nulla semper quis. Pellentesque consequat vel nunc at scelerisque. Mauris sit amet venenatis diam, blandit viverra leo. Integer commodo laoreet orci. |  | ||||||
|  |  | ||||||
| Curabitur ipsum tortor, sodales ut augue sed, commodo porttitor libero. Pellentesque molestie vitae mi consectetur tempor. In sed lectus consequat, lobortis neque non, semper ipsum. Etiam eget ex et nibh sagittis pulvinar lacinia ac mauris. Aenean ligula eros, viverra ac nibh at, venenatis semper quam. Sed interdum ligula sit amet massa tincidunt tincidunt. Suspendisse potenti. Aliquam egestas facilisis est, sed faucibus erat scelerisque id. Duis dolor quam, viverra vitae orci euismod, laoreet pellentesque justo. Nunc malesuada non erat at ullamcorper. Mauris eget posuere odio. Vestibulum turpis nunc, pharetra eget ante in, feugiat mollis justo. Proin porttitor, diam nec vulputate pretium, tellus arcu rhoncus turpis, a blandit nisi nulla quis arcu. Nunc ac ullamcorper ligula, nec facilisis leo. |  | ||||||
|  |  | ||||||
| In vitae eros sollicitudin, iaculis ex eget, egestas orci. Etiam sed pretium lorem. Nam nisi enim, consectetur sit amet semper ac, semper pharetra diam. In pulvinar neque sapien, ac ullamcorper est lacinia a. Etiam tincidunt velit sed diam malesuada, eu ornare ex consectetur. Phasellus in imperdiet tellus. Sed bibendum, dui sit amet fringilla aliquet, enim odio sollicitudin lorem, vel semper turpis mauris vel mauris. Aenean congue magna ac massa cursus, in dictum orci commodo. Pellentesque mollis velit in sollicitudin tincidunt. Vestibulum et efficitur nulla. |  | ||||||
|  |  | ||||||
| Quisque posuere, velit sed porttitor dapibus, neque augue fringilla felis, eu luctus nisi nisl nec ipsum. Curabitur pellentesque ac lectus eget ultricies. Vestibulum est dolor, lacinia pharetra vulputate a, facilisis a magna. Nam vitae arcu nibh. Praesent finibus blandit ante, ac gravida ex mollis eget. Donec quam est, pulvinar vitae neque ut, bibendum aliquam erat. Nullam mollis arcu at sem tincidunt, in tristique lectus facilisis. Aenean ut lacus vel nisl finibus iaculis non a turpis. Integer eget ipsum ante. Donec nunc neque, vestibulum ac magna ac, posuere scelerisque dui. Pellentesque massa nibh, rhoncus id dolor quis, placerat posuere turpis. Donec aliquet augue nisi, eu finibus dui auctor et. Vestibulum eu varius lorem. Quisque lectus ante, malesuada pretium risus eget, interdum mattis enim. |  | ||||||
| `; |  | ||||||
|  |  | ||||||
| const SMALL_TEXT = "Lorem ipsum dolor sit amet, consectetur adipiscing elit."; | const SMALL_TEXT = "Lorem ipsum dolor sit amet, consectetur adipiscing elit."; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,17 +1,109 @@ | |||||||
| /* eslint-disable lit/no-template-arrow */ | /* eslint-disable lit/no-template-arrow */ | ||||||
| import "@material/mwc-button"; | import "@material/mwc-button"; | ||||||
| import { LitElement, TemplateResult, html } from "lit"; | import { html, LitElement, TemplateResult } from "lit"; | ||||||
| import { customElement, state } from "lit/decorators"; | import { customElement, state } from "lit/decorators"; | ||||||
| import { computeInitialHaFormData } from "../../../../src/components/ha-form/compute-initial-ha-form-data"; |  | ||||||
| import type { HaFormSchema } from "../../../../src/components/ha-form/types"; |  | ||||||
| import "../../../../src/components/ha-form/ha-form"; |  | ||||||
| import "../../components/demo-black-white-row"; |  | ||||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||||
| import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||||
|  | import { computeInitialHaFormData } from "../../../../src/components/ha-form/compute-initial-ha-form-data"; | ||||||
|  | import "../../../../src/components/ha-form/ha-form"; | ||||||
|  | import type { HaFormSchema } from "../../../../src/components/ha-form/types"; | ||||||
|  | import { getEntity } from "../../../../src/fake_data/entity"; | ||||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||||
| import { HomeAssistant } from "../../../../src/types"; | import { HomeAssistant } from "../../../../src/types"; | ||||||
|  | import "../../components/demo-black-white-row"; | ||||||
|  |  | ||||||
|  | const ENTITIES = [ | ||||||
|  |   getEntity("alarm_control_panel", "alarm", "disarmed", { | ||||||
|  |     friendly_name: "Alarm", | ||||||
|  |   }), | ||||||
|  |   getEntity("media_player", "livingroom", "playing", { | ||||||
|  |     friendly_name: "Livingroom", | ||||||
|  |   }), | ||||||
|  |   getEntity("media_player", "lounge", "idle", { | ||||||
|  |     friendly_name: "Lounge", | ||||||
|  |     supported_features: 444983, | ||||||
|  |   }), | ||||||
|  |   getEntity("light", "bedroom", "on", { | ||||||
|  |     friendly_name: "Bedroom", | ||||||
|  |   }), | ||||||
|  |   getEntity("switch", "coffee", "off", { | ||||||
|  |     friendly_name: "Coffee", | ||||||
|  |   }), | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | const DEVICES = [ | ||||||
|  |   { | ||||||
|  |     area_id: "bedroom", | ||||||
|  |     configuration_url: null, | ||||||
|  |     config_entries: ["config_entry_1"], | ||||||
|  |     connections: [], | ||||||
|  |     disabled_by: null, | ||||||
|  |     entry_type: null, | ||||||
|  |     id: "device_1", | ||||||
|  |     identifiers: [["demo", "volume1"] as [string, string]], | ||||||
|  |     manufacturer: null, | ||||||
|  |     model: null, | ||||||
|  |     name_by_user: null, | ||||||
|  |     name: "Dishwasher", | ||||||
|  |     sw_version: null, | ||||||
|  |     hw_version: null, | ||||||
|  |     via_device_id: null, | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     area_id: "backyard", | ||||||
|  |     configuration_url: null, | ||||||
|  |     config_entries: ["config_entry_2"], | ||||||
|  |     connections: [], | ||||||
|  |     disabled_by: null, | ||||||
|  |     entry_type: null, | ||||||
|  |     id: "device_2", | ||||||
|  |     identifiers: [["demo", "pwm1"] as [string, string]], | ||||||
|  |     manufacturer: null, | ||||||
|  |     model: null, | ||||||
|  |     name_by_user: null, | ||||||
|  |     name: "Lamp", | ||||||
|  |     sw_version: null, | ||||||
|  |     hw_version: null, | ||||||
|  |     via_device_id: null, | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     area_id: null, | ||||||
|  |     configuration_url: null, | ||||||
|  |     config_entries: ["config_entry_3"], | ||||||
|  |     connections: [], | ||||||
|  |     disabled_by: null, | ||||||
|  |     entry_type: null, | ||||||
|  |     id: "device_3", | ||||||
|  |     identifiers: [["demo", "pwm1"] as [string, string]], | ||||||
|  |     manufacturer: null, | ||||||
|  |     model: null, | ||||||
|  |     name_by_user: "User name", | ||||||
|  |     name: "Technical name", | ||||||
|  |     sw_version: null, | ||||||
|  |     hw_version: null, | ||||||
|  |     via_device_id: null, | ||||||
|  |   }, | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | const AREAS = [ | ||||||
|  |   { | ||||||
|  |     area_id: "backyard", | ||||||
|  |     name: "Backyard", | ||||||
|  |     picture: null, | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     area_id: "bedroom", | ||||||
|  |     name: "Bedroom", | ||||||
|  |     picture: null, | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     area_id: "livingroom", | ||||||
|  |     name: "Livingroom", | ||||||
|  |     picture: null, | ||||||
|  |   }, | ||||||
|  | ]; | ||||||
|  |  | ||||||
| const SCHEMAS: { | const SCHEMAS: { | ||||||
|   title: string; |   title: string; | ||||||
| @@ -38,6 +130,8 @@ const SCHEMAS: { | |||||||
|       select: "Select", |       select: "Select", | ||||||
|       icon: "Icon", |       icon: "Icon", | ||||||
|       media: "Media", |       media: "Media", | ||||||
|  |       location: "Location", | ||||||
|  |       entities: "Entities", | ||||||
|     }, |     }, | ||||||
|     schema: [ |     schema: [ | ||||||
|       { name: "addon", selector: { addon: {} } }, |       { name: "addon", selector: { addon: {} } }, | ||||||
| @@ -45,6 +139,7 @@ const SCHEMAS: { | |||||||
|       { |       { | ||||||
|         name: "Attribute", |         name: "Attribute", | ||||||
|         selector: { attribute: { entity_id: "" } }, |         selector: { attribute: { entity_id: "" } }, | ||||||
|  |         context: { filter_entity: "entity" }, | ||||||
|       }, |       }, | ||||||
|       { name: "Device", selector: { device: {} } }, |       { name: "Device", selector: { device: {} } }, | ||||||
|       { name: "Duration", selector: { duration: {} } }, |       { name: "Duration", selector: { duration: {} } }, | ||||||
| @@ -52,7 +147,9 @@ const SCHEMAS: { | |||||||
|       { name: "target", selector: { target: {} } }, |       { name: "target", selector: { target: {} } }, | ||||||
|       { name: "number", selector: { number: { min: 0, max: 10 } } }, |       { name: "number", selector: { number: { min: 0, max: 10 } } }, | ||||||
|       { name: "boolean", selector: { boolean: {} } }, |       { name: "boolean", selector: { boolean: {} } }, | ||||||
|       { name: "time", selector: { time: {} } }, |       { name: "time", required: true, selector: { time: {} } }, | ||||||
|  |       { name: "datetime", required: true, selector: { datetime: {} } }, | ||||||
|  |       { name: "date", required: true, selector: { date: {} } }, | ||||||
|       { name: "action", selector: { action: {} } }, |       { name: "action", selector: { action: {} } }, | ||||||
|       { name: "text", selector: { text: { multiline: false } } }, |       { name: "text", selector: { text: { multiline: false } } }, | ||||||
|       { name: "text_multiline", selector: { text: { multiline: true } } }, |       { name: "text_multiline", selector: { text: { multiline: true } } }, | ||||||
| @@ -75,6 +172,14 @@ const SCHEMAS: { | |||||||
|           media: {}, |           media: {}, | ||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|  |       { | ||||||
|  |         name: "location", | ||||||
|  |         selector: { location: { radius: true, icon: "mdi:home" } }, | ||||||
|  |       }, | ||||||
|  |       { | ||||||
|  |         name: "entities", | ||||||
|  |         selector: { entity: { multiple: true } }, | ||||||
|  |       }, | ||||||
|     ], |     ], | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
| @@ -315,9 +420,10 @@ class DemoHaForm extends LitElement { | |||||||
|     const hass = provideHass(this); |     const hass = provideHass(this); | ||||||
|     hass.updateTranslations(null, "en"); |     hass.updateTranslations(null, "en"); | ||||||
|     hass.updateTranslations("config", "en"); |     hass.updateTranslations("config", "en"); | ||||||
|  |     hass.addEntities(ENTITIES); | ||||||
|     mockEntityRegistry(hass); |     mockEntityRegistry(hass); | ||||||
|     mockDeviceRegistry(hass); |     mockDeviceRegistry(hass, DEVICES); | ||||||
|     mockAreaRegistry(hass); |     mockAreaRegistry(hass, AREAS); | ||||||
|     mockHassioSupervisor(hass); |     mockHassioSupervisor(hass); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,20 +1,20 @@ | |||||||
| /* eslint-disable lit/no-template-arrow */ | /* eslint-disable lit/no-template-arrow */ | ||||||
| import "@material/mwc-button"; | import "@material/mwc-button"; | ||||||
| import { LitElement, TemplateResult, css, html } from "lit"; | import { css, html, LitElement, TemplateResult } from "lit"; | ||||||
| import { customElement, state } from "lit/decorators"; | import { customElement, state } from "lit/decorators"; | ||||||
|  | import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; | ||||||
|  | import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; | ||||||
|  | import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; | ||||||
|  | import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; | ||||||
| import "../../../../src/components/ha-selector/ha-selector"; | import "../../../../src/components/ha-selector/ha-selector"; | ||||||
| import "../../../../src/components/ha-settings-row"; | import "../../../../src/components/ha-settings-row"; | ||||||
|  | import { BlueprintInput } from "../../../../src/data/blueprint"; | ||||||
|  | import { showDialog } from "../../../../src/dialogs/make-dialog-manager"; | ||||||
|  | import { getEntity } from "../../../../src/fake_data/entity"; | ||||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||||
|  | import { ProvideHassElement } from "../../../../src/mixins/provide-hass-lit-mixin"; | ||||||
| import type { HomeAssistant } from "../../../../src/types"; | import type { HomeAssistant } from "../../../../src/types"; | ||||||
| import "../../components/demo-black-white-row"; | import "../../components/demo-black-white-row"; | ||||||
| import { BlueprintInput } from "../../../../src/data/blueprint"; |  | ||||||
| import { mockEntityRegistry } from "../../../../demo/src/stubs/entity_registry"; |  | ||||||
| import { mockDeviceRegistry } from "../../../../demo/src/stubs/device_registry"; |  | ||||||
| import { mockAreaRegistry } from "../../../../demo/src/stubs/area_registry"; |  | ||||||
| import { mockHassioSupervisor } from "../../../../demo/src/stubs/hassio_supervisor"; |  | ||||||
| import { getEntity } from "../../../../src/fake_data/entity"; |  | ||||||
| import { ProvideHassElement } from "../../../../src/mixins/provide-hass-lit-mixin"; |  | ||||||
| import { showDialog } from "../../../../src/dialogs/make-dialog-manager"; |  | ||||||
|  |  | ||||||
| const ENTITIES = [ | const ENTITIES = [ | ||||||
|   getEntity("alarm_control_panel", "alarm", "disarmed", { |   getEntity("alarm_control_panel", "alarm", "disarmed", { | ||||||
| @@ -109,7 +109,7 @@ const AREAS = [ | |||||||
|  |  | ||||||
| const SCHEMAS: { | const SCHEMAS: { | ||||||
|   name: string; |   name: string; | ||||||
|   input: Record<string, BlueprintInput | null>; |   input: Record<string, (BlueprintInput & { required?: boolean }) | null>; | ||||||
| }[] = [ | }[] = [ | ||||||
|   { |   { | ||||||
|     name: "One of each", |     name: "One of each", | ||||||
| @@ -146,6 +146,8 @@ const SCHEMAS: { | |||||||
|       }, |       }, | ||||||
|       boolean: { name: "Boolean", selector: { boolean: {} } }, |       boolean: { name: "Boolean", selector: { boolean: {} } }, | ||||||
|       time: { name: "Time", selector: { time: {} } }, |       time: { name: "Time", selector: { time: {} } }, | ||||||
|  |       date: { name: "Date", selector: { date: {} } }, | ||||||
|  |       datetime: { name: "Date Time", selector: { datetime: {} } }, | ||||||
|       action: { name: "Action", selector: { action: {} } }, |       action: { name: "Action", selector: { action: {} } }, | ||||||
|       text: { |       text: { | ||||||
|         name: "Text", |         name: "Text", | ||||||
| @@ -162,12 +164,92 @@ const SCHEMAS: { | |||||||
|         }, |         }, | ||||||
|       }, |       }, | ||||||
|       object: { name: "Object", selector: { object: {} } }, |       object: { name: "Object", selector: { object: {} } }, | ||||||
|  |       select_radio: { | ||||||
|  |         name: "Select (Radio)", | ||||||
|  |         selector: { | ||||||
|  |           select: { options: ["Option 1", "Option 2"], mode: "list" }, | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |       template: { name: "Template", selector: { template: {} } }, | ||||||
|       select: { |       select: { | ||||||
|         name: "Select", |         name: "Select", | ||||||
|         selector: { select: { options: ["Option 1", "Option 2"] } }, |         selector: { | ||||||
|  |           select: { | ||||||
|  |             options: [ | ||||||
|  |               "Option 1", | ||||||
|  |               "Option 2", | ||||||
|  |               "Option 3", | ||||||
|  |               "Option 4", | ||||||
|  |               "Option 5", | ||||||
|  |               "Option 6", | ||||||
|  |             ], | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |       select_custom: { | ||||||
|  |         name: "Select (Custom)", | ||||||
|  |         selector: { | ||||||
|  |           select: { | ||||||
|  |             custom_value: true, | ||||||
|  |             options: [ | ||||||
|  |               "Option 1", | ||||||
|  |               "Option 2", | ||||||
|  |               "Option 3", | ||||||
|  |               "Option 4", | ||||||
|  |               "Option 5", | ||||||
|  |               "Option 6", | ||||||
|  |             ], | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|       }, |       }, | ||||||
|       icon: { name: "Icon", selector: { icon: {} } }, |       icon: { name: "Icon", selector: { icon: {} } }, | ||||||
|       media: { name: "Media", selector: { media: {} } }, |       media: { name: "Media", selector: { media: {} } }, | ||||||
|  |       location: { name: "Location", selector: { location: {} } }, | ||||||
|  |       location_radius: { | ||||||
|  |         name: "Location with radius", | ||||||
|  |         selector: { location: { radius: true, icon: "mdi:home" } }, | ||||||
|  |       }, | ||||||
|  |       color_temp: { | ||||||
|  |         name: "Color Temperature", | ||||||
|  |         selector: { color_temp: {} }, | ||||||
|  |       }, | ||||||
|  |       color_rgb: { name: "Color", selector: { color_rgb: {} } }, | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     name: "Multiples", | ||||||
|  |     input: { | ||||||
|  |       entity: { name: "Entity", selector: { entity: { multiple: true } } }, | ||||||
|  |       device: { name: "Device", selector: { device: { multiple: true } } }, | ||||||
|  |       area: { name: "Area", selector: { area: { multiple: true } } }, | ||||||
|  |       select: { | ||||||
|  |         name: "Select Multiple", | ||||||
|  |         selector: { | ||||||
|  |           select: { | ||||||
|  |             multiple: true, | ||||||
|  |             custom_value: true, | ||||||
|  |             options: [ | ||||||
|  |               "Option 1", | ||||||
|  |               "Option 2", | ||||||
|  |               "Option 3", | ||||||
|  |               "Option 4", | ||||||
|  |               "Option 5", | ||||||
|  |               "Option 6", | ||||||
|  |             ], | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|  |       select_checkbox: { | ||||||
|  |         name: "Select Multiple (Checkbox)", | ||||||
|  |         required: false, | ||||||
|  |         selector: { | ||||||
|  |           select: { | ||||||
|  |             mode: "list", | ||||||
|  |             multiple: true, | ||||||
|  |             options: ["Option 1", "Option 2", "Option 3", "Option 4"], | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       }, | ||||||
|     }, |     }, | ||||||
|   }, |   }, | ||||||
| ]; | ]; | ||||||
| @@ -176,6 +258,14 @@ const SCHEMAS: { | |||||||
| class DemoHaSelector extends LitElement implements ProvideHassElement { | class DemoHaSelector extends LitElement implements ProvideHassElement { | ||||||
|   @state() public hass!: HomeAssistant; |   @state() public hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  |   @state() private _disabled = false; | ||||||
|  |  | ||||||
|  |   @state() private _required = false; | ||||||
|  |  | ||||||
|  |   @state() private _helper = false; | ||||||
|  |  | ||||||
|  |   @state() private _label = true; | ||||||
|  |  | ||||||
|   private data = SCHEMAS.map(() => ({})); |   private data = SCHEMAS.map(() => ({})); | ||||||
|  |  | ||||||
|   constructor() { |   constructor() { | ||||||
| @@ -279,7 +369,7 @@ class DemoHaSelector extends LitElement implements ProvideHassElement { | |||||||
|           can_play: true, |           can_play: true, | ||||||
|           can_expand: false, |           can_expand: false, | ||||||
|           children_media_class: null, |           children_media_class: null, | ||||||
|           thumbnail: null, |           thumbnail: "https://brands.home-assistant.io/_/image/logo.png", | ||||||
|         }, |         }, | ||||||
|         { |         { | ||||||
|           title: "movie.mp4", |           title: "movie.mp4", | ||||||
| @@ -309,6 +399,36 @@ class DemoHaSelector extends LitElement implements ProvideHassElement { | |||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     return html` |     return html` | ||||||
|  |       <div class="options"> | ||||||
|  |         <ha-formfield label="Labels"> | ||||||
|  |           <ha-switch | ||||||
|  |             .name=${"label"} | ||||||
|  |             .checked=${this._label} | ||||||
|  |             @change=${this._handleOptionChange} | ||||||
|  |           ></ha-switch> | ||||||
|  |         </ha-formfield> | ||||||
|  |         <ha-formfield label="Required"> | ||||||
|  |           <ha-switch | ||||||
|  |             .name=${"required"} | ||||||
|  |             .checked=${this._required} | ||||||
|  |             @change=${this._handleOptionChange} | ||||||
|  |           ></ha-switch> | ||||||
|  |         </ha-formfield> | ||||||
|  |         <ha-formfield label="Disabled"> | ||||||
|  |           <ha-switch | ||||||
|  |             .name=${"disabled"} | ||||||
|  |             .checked=${this._disabled} | ||||||
|  |             @change=${this._handleOptionChange} | ||||||
|  |           ></ha-switch> | ||||||
|  |         </ha-formfield> | ||||||
|  |         <ha-formfield label="Helper text"> | ||||||
|  |           <ha-switch | ||||||
|  |             .name=${"helper"} | ||||||
|  |             .checked=${this._helper} | ||||||
|  |             @change=${this._handleOptionChange} | ||||||
|  |           ></ha-switch> | ||||||
|  |         </ha-formfield> | ||||||
|  |       </div> | ||||||
|       ${SCHEMAS.map((info, idx) => { |       ${SCHEMAS.map((info, idx) => { | ||||||
|         const data = this.data[idx]; |         const data = this.data[idx]; | ||||||
|         const valueChanged = (ev) => { |         const valueChanged = (ev) => { | ||||||
| @@ -331,8 +451,12 @@ class DemoHaSelector extends LitElement implements ProvideHassElement { | |||||||
|                         .hass=${this.hass} |                         .hass=${this.hass} | ||||||
|                         .selector=${value!.selector} |                         .selector=${value!.selector} | ||||||
|                         .key=${key} |                         .key=${key} | ||||||
|  |                         .label=${this._label ? value!.name : undefined} | ||||||
|                         .value=${data[key] ?? value!.default} |                         .value=${data[key] ?? value!.default} | ||||||
|  |                         .disabled=${this._disabled} | ||||||
|  |                         .required=${this._required} | ||||||
|                         @value-changed=${valueChanged} |                         @value-changed=${valueChanged} | ||||||
|  |                         .helper=${this._helper ? "Helper text" : undefined} | ||||||
|                       ></ha-selector> |                       ></ha-selector> | ||||||
|                     </ha-settings-row> |                     </ha-settings-row> | ||||||
|                   ` |                   ` | ||||||
| @@ -344,10 +468,21 @@ class DemoHaSelector extends LitElement implements ProvideHassElement { | |||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private _handleOptionChange(ev) { | ||||||
|  |     this[`_${ev.target.name}`] = ev.target.checked; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   static styles = css` |   static styles = css` | ||||||
|     ha-selector { |     ha-selector { | ||||||
|       width: 60; |       width: 60; | ||||||
|     } |     } | ||||||
|  |     .options { | ||||||
|  |       max-width: 800px; | ||||||
|  |       margin: 16px auto; | ||||||
|  |     } | ||||||
|  |     .options ha-formfield { | ||||||
|  |       margin-right: 16px; | ||||||
|  |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								gallery/src/pages/components/ha-tip.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/components/ha-tip.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | --- | ||||||
|  | title: Tips | ||||||
|  | --- | ||||||
							
								
								
									
										73
									
								
								gallery/src/pages/components/ha-tip.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								gallery/src/pages/components/ha-tip.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,73 @@ | |||||||
|  | import { html, css, LitElement, TemplateResult } from "lit"; | ||||||
|  | import { customElement } from "lit/decorators"; | ||||||
|  | import "../../../../src/components/ha-tip"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||||
|  |  | ||||||
|  | const tips: (string | TemplateResult)[] = [ | ||||||
|  |   "Test tip", | ||||||
|  |   "Bigger test tip, with some random text just to fill up as much space as possible without it looking like I'm really trying to to that", | ||||||
|  |   html`<i>Tip</i> <b>with</b> <sub>HTML</sub>`, | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | @customElement("demo-components-ha-tip") | ||||||
|  | export class DemoHaTip extends LitElement { | ||||||
|  |   protected render(): TemplateResult { | ||||||
|  |     return html` ${["light", "dark"].map( | ||||||
|  |       (mode) => html` | ||||||
|  |         <div class=${mode}> | ||||||
|  |           <ha-card header="ha-tip ${mode} demo"> | ||||||
|  |             <div class="card-content"> | ||||||
|  |               ${tips.map((tip) => html`<ha-tip>${tip}</ha-tip>`)} | ||||||
|  |             </div> | ||||||
|  |           </ha-card> | ||||||
|  |         </div> | ||||||
|  |       ` | ||||||
|  |     )}`; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   firstUpdated(changedProps) { | ||||||
|  |     super.firstUpdated(changedProps); | ||||||
|  |     applyThemesOnElement( | ||||||
|  |       this.shadowRoot!.querySelector(".dark"), | ||||||
|  |       { | ||||||
|  |         default_theme: "default", | ||||||
|  |         default_dark_theme: "default", | ||||||
|  |         themes: {}, | ||||||
|  |         darkMode: true, | ||||||
|  |         theme: "default", | ||||||
|  |       }, | ||||||
|  |       undefined, | ||||||
|  |       undefined, | ||||||
|  |       true | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static get styles() { | ||||||
|  |     return css` | ||||||
|  |       :host { | ||||||
|  |         display: flex; | ||||||
|  |         flex-direction: row; | ||||||
|  |         justify-content: space-between; | ||||||
|  |       } | ||||||
|  |       .dark, | ||||||
|  |       .light { | ||||||
|  |         display: block; | ||||||
|  |         background-color: var(--primary-background-color); | ||||||
|  |         padding: 0 50px; | ||||||
|  |       } | ||||||
|  |       ha-tip { | ||||||
|  |         margin-bottom: 14px; | ||||||
|  |       } | ||||||
|  |       ha-card { | ||||||
|  |         margin: 24px auto; | ||||||
|  |       } | ||||||
|  |     `; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-components-ha-tip": DemoHaTip; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -2,6 +2,8 @@ | |||||||
| title: Editing design.home-assistant.io | title: Editing design.home-assistant.io | ||||||
| --- | --- | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # How to edit design.home-assistant.io | # How to edit design.home-assistant.io | ||||||
|  |  | ||||||
| All pages are stored in [the pages folder][pages-folder] on GitHub. Pages are grouped in a folder per sidebar section. Each page can contain a `<page name>.markdown` description file, a `<page name>.ts` demo file or both. If both are defined the description is rendered first. The description can contain metadata to specify the title of the page. | All pages are stored in [the pages folder][pages-folder] on GitHub. Pages are grouped in a folder per sidebar section. Each page can contain a `<page name>.markdown` description file, a `<page name>.ts` demo file or both. If both are defined the description is rendered first. The description can contain metadata to specify the title of the page. | ||||||
| @@ -41,15 +43,12 @@ import { html, css, LitElement } from "lit"; | |||||||
| import { customElement } from "lit/decorators"; | import { customElement } from "lit/decorators"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
|  |  | ||||||
|  |  | ||||||
| @customElement("demo-user-experience-usability") | @customElement("demo-user-experience-usability") | ||||||
| export class DemoUserExperienceUsability extends LitElement { | export class DemoUserExperienceUsability extends LitElement { | ||||||
|   protected render() { |   protected render() { | ||||||
|     return html` |     return html` | ||||||
|       <ha-card> |       <ha-card> | ||||||
|         <div class="card-content"> |         <div class="card-content">Hello world!</div> | ||||||
|           Hello world! |  | ||||||
|         </div> |  | ||||||
|       </ha-card> |       </ha-card> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -249,7 +249,7 @@ const CONFIGS = [ | |||||||
|       name: Bed light |       name: Bed light | ||||||
|       action_name: Toggle light |       action_name: Toggle light | ||||||
|       service: light.toggle |       service: light.toggle | ||||||
|       service_data: |       data: | ||||||
|         entity_id: light.bed_light |         entity_id: light.bed_light | ||||||
|     - type: section |     - type: section | ||||||
|       label: Links |       label: Links | ||||||
|   | |||||||
| @@ -199,7 +199,7 @@ const CONFIGS = [ | |||||||
|       tap_action: |       tap_action: | ||||||
|         action: call-service |         action: call-service | ||||||
|         service: light.turn_on |         service: light.turn_on | ||||||
|         service_data: |         data: | ||||||
|           entity_id: light.ceiling_lights |           entity_id: light.ceiling_lights | ||||||
|     - entity: sun.sun |     - entity: sun.sun | ||||||
|       name: Regular |       name: Regular | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ const CONFIGS = [ | |||||||
|     heading: "markdown-it demo", |     heading: "markdown-it demo", | ||||||
|     config: ` |     config: ` | ||||||
| - type: markdown | - type: markdown | ||||||
|   content: > |   content: >- | ||||||
|     # h1 Heading 8-) |     # h1 Heading 8-) | ||||||
|  |  | ||||||
|     ## h2 Heading |     ## h2 Heading | ||||||
| @@ -249,6 +249,17 @@ const CONFIGS = [ | |||||||
|     ::: warning |     ::: warning | ||||||
|     *here be dragons* |     *here be dragons* | ||||||
|     ::: |     ::: | ||||||
|  |  | ||||||
|  |     ### ha-alert | ||||||
|  |  | ||||||
|  |     You can use our [\`ha-alert\`](https://design.home-assistant.io/#components/ha-alert) component in markdown content rendered in the Home Assistant Frontend. | ||||||
|  |  | ||||||
|  |     <ha-alert alert-type="error">This is an error alert — check it out!</ha-alert> | ||||||
|  |     <ha-alert alert-type="warning">This is a warning alert — check it out!</ha-alert> | ||||||
|  |     <ha-alert alert-type="info">This is an info alert — check it out!</ha-alert> | ||||||
|  |     <ha-alert alert-type="success">This is a success alert — check it out!</ha-alert> | ||||||
|  |     <ha-alert title="Test alert">This is an alert with a title</ha-alert> | ||||||
|  |  | ||||||
|     `, |     `, | ||||||
|   }, |   }, | ||||||
| ]; | ]; | ||||||
|   | |||||||
| @@ -40,7 +40,7 @@ const CONFIGS = [ | |||||||
|         left: 90% |         left: 90% | ||||||
|         padding: 0px |         padding: 0px | ||||||
|       service: light.turn_off |       service: light.turn_off | ||||||
|       service_data: |       data: | ||||||
|         entity_id: group.all_lights |         entity_id: group.all_lights | ||||||
|     - type: icon |     - type: icon | ||||||
|       icon: mdi:cctv |       icon: mdi:cctv | ||||||
| @@ -88,7 +88,7 @@ const CONFIGS = [ | |||||||
|         left: 90% |         left: 90% | ||||||
|         padding: 0px |         padding: 0px | ||||||
|       service: light.turn_off |       service: light.turn_off | ||||||
|       service_data: |       data: | ||||||
|         entity_id: group.all_lights |         entity_id: group.all_lights | ||||||
|     - type: icon |     - type: icon | ||||||
|       icon: mdi:cctv |       icon: mdi:cctv | ||||||
|   | |||||||
| @@ -188,6 +188,7 @@ const createEntityRegistryEntries = ( | |||||||
|     device_id: "mock-device-id", |     device_id: "mock-device-id", | ||||||
|     area_id: null, |     area_id: null, | ||||||
|     disabled_by: null, |     disabled_by: null, | ||||||
|  |     hidden_by: null, | ||||||
|     entity_category: null, |     entity_category: null, | ||||||
|     entity_id: "binary_sensor.updater", |     entity_id: "binary_sensor.updater", | ||||||
|     name: null, |     name: null, | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								gallery/src/pages/more-info/update.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/more-info/update.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | --- | ||||||
|  | title: Update | ||||||
|  | --- | ||||||
							
								
								
									
										189
									
								
								gallery/src/pages/more-info/update.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										189
									
								
								gallery/src/pages/more-info/update.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,189 @@ | |||||||
|  | import { html, LitElement, PropertyValues, TemplateResult } from "lit"; | ||||||
|  | import { customElement, property, query } from "lit/decorators"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import { | ||||||
|  |   UPDATE_SUPPORT_BACKUP, | ||||||
|  |   UPDATE_SUPPORT_PROGRESS, | ||||||
|  |   UPDATE_SUPPORT_INSTALL, | ||||||
|  |   UPDATE_SUPPORT_RELEASE_NOTES, | ||||||
|  | } from "../../../../src/data/update"; | ||||||
|  | import "../../../../src/dialogs/more-info/more-info-content"; | ||||||
|  | import { getEntity } from "../../../../src/fake_data/entity"; | ||||||
|  | import { | ||||||
|  |   MockHomeAssistant, | ||||||
|  |   provideHass, | ||||||
|  | } from "../../../../src/fake_data/provide_hass"; | ||||||
|  | import "../../components/demo-more-infos"; | ||||||
|  | import { LONG_TEXT } from "../../data/text"; | ||||||
|  |  | ||||||
|  | const base_attributes = { | ||||||
|  |   title: "Awesome", | ||||||
|  |   installed_version: "1.2.2", | ||||||
|  |   latest_version: "1.2.3", | ||||||
|  |   release_url: "https://home-assistant.io", | ||||||
|  |   supported_features: UPDATE_SUPPORT_INSTALL, | ||||||
|  |   skipped_version: null, | ||||||
|  |   in_progress: false, | ||||||
|  |   release_summary: | ||||||
|  |     "Lorem ipsum dolor sit amet, consectetur adipiscing elit. In nec metus aliquet, porta mi ut, ultrices odio. Etiam egestas orci tellus, non semper metus blandit tincidunt. Praesent elementum turpis vel tempor pharetra. Sed quis cursus diam. Proin sem justo.", | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const ENTITIES = [ | ||||||
|  |   getEntity("update", "update1", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     friendly_name: "Update", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update2", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     title: null, | ||||||
|  |     friendly_name: "Update without title", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update3", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     release_url: null, | ||||||
|  |     friendly_name: "Update without release_url", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update4", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     release_summary: null, | ||||||
|  |     friendly_name: "Update without release_summary", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update5", "off", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     installed_version: "1.2.3", | ||||||
|  |     friendly_name: "No update", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update6", "off", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     skipped_version: "1.2.3", | ||||||
|  |     friendly_name: "Skipped version", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update7", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     supported_features: | ||||||
|  |       base_attributes.supported_features + UPDATE_SUPPORT_BACKUP, | ||||||
|  |     friendly_name: "With backup support", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update8", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     in_progress: true, | ||||||
|  |     friendly_name: "With true in_progress", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update9", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     in_progress: 25, | ||||||
|  |     supported_features: | ||||||
|  |       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||||
|  |     friendly_name: "With 25 in_progress", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update10", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     in_progress: 50, | ||||||
|  |     supported_features: | ||||||
|  |       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||||
|  |     friendly_name: "With 50 in_progress", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update11", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     in_progress: 75, | ||||||
|  |     supported_features: | ||||||
|  |       base_attributes.supported_features + UPDATE_SUPPORT_PROGRESS, | ||||||
|  |     friendly_name: "With 75 in_progress", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update12", "unavailable", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     in_progress: 50, | ||||||
|  |     friendly_name: "Unavailable", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update13", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     supported_features: 0, | ||||||
|  |     friendly_name: "No install support", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update14", "off", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     installed_version: null, | ||||||
|  |     friendly_name: "Update without installed_version", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update15", "off", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     latest_version: null, | ||||||
|  |     friendly_name: "Update without latest_version", | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update16", "off", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     friendly_name: "Update with release notes", | ||||||
|  |     supported_features: | ||||||
|  |       base_attributes.supported_features + UPDATE_SUPPORT_RELEASE_NOTES, | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update17", "off", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     friendly_name: "Update with release notes error", | ||||||
|  |     supported_features: | ||||||
|  |       base_attributes.supported_features + UPDATE_SUPPORT_RELEASE_NOTES, | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update18", "off", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     friendly_name: "Update with release notes loading", | ||||||
|  |     supported_features: | ||||||
|  |       base_attributes.supported_features + UPDATE_SUPPORT_RELEASE_NOTES, | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update19", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     friendly_name: "Update with auto update", | ||||||
|  |     auto_update: true, | ||||||
|  |   }), | ||||||
|  |   getEntity("update", "update20", "on", { | ||||||
|  |     ...base_attributes, | ||||||
|  |     in_progress: true, | ||||||
|  |     title: undefined, | ||||||
|  |     friendly_name: "Installing without title", | ||||||
|  |   }), | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | @customElement("demo-more-info-update") | ||||||
|  | class DemoMoreInfoUpdate extends LitElement { | ||||||
|  |   @property() public hass!: MockHomeAssistant; | ||||||
|  |  | ||||||
|  |   @query("demo-more-infos") private _demoRoot!: HTMLElement; | ||||||
|  |  | ||||||
|  |   protected render(): TemplateResult { | ||||||
|  |     return html` | ||||||
|  |       <demo-more-infos | ||||||
|  |         .hass=${this.hass} | ||||||
|  |         .entities=${ENTITIES.map((ent) => ent.entityId)} | ||||||
|  |       ></demo-more-infos> | ||||||
|  |     `; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected firstUpdated(changedProperties: PropertyValues) { | ||||||
|  |     super.firstUpdated(changedProperties); | ||||||
|  |     const hass = provideHass(this._demoRoot); | ||||||
|  |     hass.updateTranslations(null, "en"); | ||||||
|  |     hass.addEntities(ENTITIES); | ||||||
|  |     hass.mockWS( | ||||||
|  |       "update/release_notes", | ||||||
|  |       (msg: { type: string; entity_id: string }) => { | ||||||
|  |         if (msg.entity_id === "update.update16") { | ||||||
|  |           return LONG_TEXT; | ||||||
|  |         } | ||||||
|  |         if (msg.entity_id === "update.update17") { | ||||||
|  |           return Promise.reject({ | ||||||
|  |             code: "error", | ||||||
|  |             message: "Could not fetch release notes", | ||||||
|  |           }); | ||||||
|  |         } | ||||||
|  |         if (msg.entity_id === "update.update18") { | ||||||
|  |           return undefined; | ||||||
|  |         } | ||||||
|  |         return null; | ||||||
|  |       } | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-more-info-update": DemoMoreInfoUpdate; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								gallery/src/pages/user-test/user-types.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								gallery/src/pages/user-test/user-types.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | --- | ||||||
|  | title: "User types" | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | We have defined three user types for Home Assistant. They are a lean segmentation of users that helps us make decisions throughout the product. User types differ from traditional personas in that the segmentation criteria aren’t demographic and don’t personify a group into a single character with a fictitious background story.  | ||||||
|  |  | ||||||
|  | # Outgrowers | ||||||
|  |  | ||||||
|  | Users that outgrow big tech smart home solutions. It just needs to work with easy setup via an app. | ||||||
|  |  | ||||||
|  | # Tinkerers | ||||||
|  |  | ||||||
|  | Technoid users in home networking and development that know how to code. | ||||||
|  |  | ||||||
|  | # Questioner | ||||||
|  |  | ||||||
|  | Users who want more advanced home automation, but need support to make it work. | ||||||
| @@ -68,6 +68,7 @@ class HassioAddonRepositoryEl extends LitElement { | |||||||
|           ${addons.map( |           ${addons.map( | ||||||
|             (addon) => html` |             (addon) => html` | ||||||
|               <ha-card |               <ha-card | ||||||
|  |                 outlined | ||||||
|                 .addon=${addon} |                 .addon=${addon} | ||||||
|                 class=${addon.available ? "" : "not_available"} |                 class=${addon.available ? "" : "not_available"} | ||||||
|                 @click=${this._addonTapped} |                 @click=${this._addonTapped} | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ import memoizeOne from "memoize-one"; | |||||||
| import { atLeastVersion } from "../../../src/common/config/version"; | import { atLeastVersion } from "../../../src/common/config/version"; | ||||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||||
| import { navigate } from "../../../src/common/navigate"; | import { navigate } from "../../../src/common/navigate"; | ||||||
| import "../../../src/common/search/search-input"; | import "../../../src/components/search-input"; | ||||||
| import { extractSearchParam } from "../../../src/common/url/search-params"; | import { extractSearchParam } from "../../../src/common/url/search-params"; | ||||||
| import "../../../src/components/ha-button-menu"; | import "../../../src/components/ha-button-menu"; | ||||||
| import "../../../src/components/ha-icon-button"; | import "../../../src/components/ha-icon-button"; | ||||||
| @@ -110,8 +110,6 @@ class HassioAddonStore extends LitElement { | |||||||
|               <div class="search"> |               <div class="search"> | ||||||
|                 <search-input |                 <search-input | ||||||
|                   .hass=${this.hass} |                   .hass=${this.hass} | ||||||
|                   no-label-float |  | ||||||
|                   no-underline |  | ||||||
|                   .filter=${this._filter} |                   .filter=${this._filter} | ||||||
|                   @value-changed=${this._filterChanged} |                   @value-changed=${this._filterChanged} | ||||||
|                 ></search-input> |                 ></search-input> | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| import "@material/mwc-button"; | import "@material/mwc-button"; | ||||||
| import "@material/mwc-select"; |  | ||||||
| import "@material/mwc-list/mwc-list-item"; | import "@material/mwc-list/mwc-list-item"; | ||||||
| import { | import { | ||||||
|   css, |   css, | ||||||
| @@ -14,6 +13,7 @@ import { stopPropagation } from "../../../../src/common/dom/stop_propagation"; | |||||||
| import "../../../../src/components/buttons/ha-progress-button"; | import "../../../../src/components/buttons/ha-progress-button"; | ||||||
| import "../../../../src/components/ha-alert"; | import "../../../../src/components/ha-alert"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-select"; | ||||||
| import { | import { | ||||||
|   HassioAddonDetails, |   HassioAddonDetails, | ||||||
|   HassioAddonSetOptionParams, |   HassioAddonSetOptionParams, | ||||||
| @@ -50,6 +50,7 @@ class HassioAddonAudio extends LitElement { | |||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     return html` |     return html` | ||||||
|       <ha-card |       <ha-card | ||||||
|  |         outlined | ||||||
|         .header=${this.supervisor.localize("addon.configuration.audio.header")} |         .header=${this.supervisor.localize("addon.configuration.audio.header")} | ||||||
|       > |       > | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
| @@ -57,7 +58,7 @@ class HassioAddonAudio extends LitElement { | |||||||
|             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` |             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||||
|             : ""} |             : ""} | ||||||
|           ${this._inputDevices && |           ${this._inputDevices && | ||||||
|           html`<mwc-select |           html`<ha-select | ||||||
|             .label=${this.supervisor.localize( |             .label=${this.supervisor.localize( | ||||||
|               "addon.configuration.audio.input" |               "addon.configuration.audio.input" | ||||||
|             )} |             )} | ||||||
| @@ -74,9 +75,9 @@ class HassioAddonAudio extends LitElement { | |||||||
|                 </mwc-list-item> |                 </mwc-list-item> | ||||||
|               ` |               ` | ||||||
|             )} |             )} | ||||||
|           </mwc-select>`} |           </ha-select>`} | ||||||
|           ${this._outputDevices && |           ${this._outputDevices && | ||||||
|           html`<mwc-select |           html`<ha-select | ||||||
|             .label=${this.supervisor.localize( |             .label=${this.supervisor.localize( | ||||||
|               "addon.configuration.audio.output" |               "addon.configuration.audio.output" | ||||||
|             )} |             )} | ||||||
| @@ -93,7 +94,7 @@ class HassioAddonAudio extends LitElement { | |||||||
|                 > |                 > | ||||||
|               ` |               ` | ||||||
|             )} |             )} | ||||||
|           </mwc-select>`} |           </ha-select>`} | ||||||
|         </div> |         </div> | ||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           <ha-progress-button @click=${this._saveSettings}> |           <ha-progress-button @click=${this._saveSettings}> | ||||||
| @@ -119,10 +120,10 @@ class HassioAddonAudio extends LitElement { | |||||||
|         .card-actions { |         .card-actions { | ||||||
|           text-align: right; |           text-align: right; | ||||||
|         } |         } | ||||||
|         mwc-select { |         ha-select { | ||||||
|           width: 100%; |           width: 100%; | ||||||
|         } |         } | ||||||
|         mwc-select:last-child { |         ha-select:last-child { | ||||||
|           margin-top: 8px; |           margin-top: 8px; | ||||||
|         } |         } | ||||||
|       `, |       `, | ||||||
|   | |||||||
| @@ -39,7 +39,14 @@ import type { HomeAssistant } from "../../../../src/types"; | |||||||
| import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart"; | import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart"; | ||||||
| import { hassioStyle } from "../../resources/hassio-style"; | import { hassioStyle } from "../../resources/hassio-style"; | ||||||
|  |  | ||||||
| const SUPPORTED_UI_TYPES = ["string", "select", "boolean", "integer", "float"]; | const SUPPORTED_UI_TYPES = [ | ||||||
|  |   "string", | ||||||
|  |   "select", | ||||||
|  |   "boolean", | ||||||
|  |   "integer", | ||||||
|  |   "float", | ||||||
|  |   "schema", | ||||||
|  | ]; | ||||||
|  |  | ||||||
| const ADDON_YAML_SCHEMA = DEFAULT_SCHEMA.extend([ | const ADDON_YAML_SCHEMA = DEFAULT_SCHEMA.extend([ | ||||||
|   new Type("!secret", { |   new Type("!secret", { | ||||||
| @@ -48,6 +55,8 @@ const ADDON_YAML_SCHEMA = DEFAULT_SCHEMA.extend([ | |||||||
|   }), |   }), | ||||||
| ]); | ]); | ||||||
|  |  | ||||||
|  | const MASKED_FIELDS = ["password", "secret", "token"]; | ||||||
|  |  | ||||||
| @customElement("hassio-addon-config") | @customElement("hassio-addon-config") | ||||||
| class HassioAddonConfig extends LitElement { | class HassioAddonConfig extends LitElement { | ||||||
|   @property({ attribute: false }) public addon!: HassioAddonDetails; |   @property({ attribute: false }) public addon!: HassioAddonDetails; | ||||||
| @@ -75,16 +84,63 @@ class HassioAddonConfig extends LitElement { | |||||||
|   public computeLabel = (entry: HaFormSchema): string => |   public computeLabel = (entry: HaFormSchema): string => | ||||||
|     this.addon.translations[this.hass.language]?.configuration?.[entry.name] |     this.addon.translations[this.hass.language]?.configuration?.[entry.name] | ||||||
|       ?.name || |       ?.name || | ||||||
|     this.addon.translations.en?.configuration?.[entry.name].name || |     this.addon.translations.en?.configuration?.[entry.name]?.name || | ||||||
|     entry.name; |     entry.name; | ||||||
|  |  | ||||||
|   private _schema = memoizeOne((schema: HaFormSchema[]): HaFormSchema[] => |   public computeHelper = (entry: HaFormSchema): string => | ||||||
|     // @ts-expect-error supervisor does not implement [string, string] for select.options[] |     this.addon.translations[this.hass.language]?.configuration?.[entry.name] | ||||||
|  |       ?.description || | ||||||
|  |     this.addon.translations.en?.configuration?.[entry.name]?.description || | ||||||
|  |     ""; | ||||||
|  |  | ||||||
|  |   private _convertSchema = memoizeOne( | ||||||
|  |     // Convert supervisor schema to selectors | ||||||
|  |     (schema: Record<string, any>): HaFormSchema[] => | ||||||
|       schema.map((entry) => |       schema.map((entry) => | ||||||
|         entry.type === "select" |         entry.type === "select" | ||||||
|           ? { |           ? { | ||||||
|             ...entry, |               name: entry.name, | ||||||
|             options: entry.options.map((option) => [option, option]), |               required: entry.required, | ||||||
|  |               selector: { select: { options: entry.options } }, | ||||||
|  |             } | ||||||
|  |           : entry.type === "string" | ||||||
|  |           ? entry.multiple | ||||||
|  |             ? { | ||||||
|  |                 name: entry.name, | ||||||
|  |                 required: entry.required, | ||||||
|  |                 selector: { | ||||||
|  |                   select: { options: [], multiple: true, custom_value: true }, | ||||||
|  |                 }, | ||||||
|  |               } | ||||||
|  |             : { | ||||||
|  |                 name: entry.name, | ||||||
|  |                 required: entry.required, | ||||||
|  |                 selector: { | ||||||
|  |                   text: { | ||||||
|  |                     type: | ||||||
|  |                       entry.format || MASKED_FIELDS.includes(entry.name) | ||||||
|  |                         ? "password" | ||||||
|  |                         : "text", | ||||||
|  |                   }, | ||||||
|  |                 }, | ||||||
|  |               } | ||||||
|  |           : entry.type === "boolean" | ||||||
|  |           ? { | ||||||
|  |               name: entry.name, | ||||||
|  |               required: entry.required, | ||||||
|  |               selector: { boolean: {} }, | ||||||
|  |             } | ||||||
|  |           : entry.type === "schema" | ||||||
|  |           ? { | ||||||
|  |               name: entry.name, | ||||||
|  |               required: entry.required, | ||||||
|  |               selector: { object: {} }, | ||||||
|  |             } | ||||||
|  |           : entry.type === "float" || entry.type === "integer" | ||||||
|  |           ? { | ||||||
|  |               name: entry.name, | ||||||
|  |               required: entry.required, | ||||||
|  |               selector: { number: { mode: "box" } }, | ||||||
|             } |             } | ||||||
|           : entry |           : entry | ||||||
|       ) |       ) | ||||||
| @@ -106,7 +162,7 @@ class HassioAddonConfig extends LitElement { | |||||||
|         ); |         ); | ||||||
|     return html` |     return html` | ||||||
|       <h1>${this.addon.name}</h1> |       <h1>${this.addon.name}</h1> | ||||||
|       <ha-card> |       <ha-card outlined> | ||||||
|         <div class="header"> |         <div class="header"> | ||||||
|           <h2> |           <h2> | ||||||
|             ${this.supervisor.localize("addon.configuration.options.header")} |             ${this.supervisor.localize("addon.configuration.options.header")} | ||||||
| @@ -140,7 +196,8 @@ class HassioAddonConfig extends LitElement { | |||||||
|                 .data=${this._options!} |                 .data=${this._options!} | ||||||
|                 @value-changed=${this._configChanged} |                 @value-changed=${this._configChanged} | ||||||
|                 .computeLabel=${this.computeLabel} |                 .computeLabel=${this.computeLabel} | ||||||
|                 .schema=${this._schema( |                 .computeHelper=${this.computeHelper} | ||||||
|  |                 .schema=${this._convertSchema( | ||||||
|                   this._showOptional |                   this._showOptional | ||||||
|                     ? this.addon.schema! |                     ? this.addon.schema! | ||||||
|                     : this._filteredShchema( |                     : this._filteredShchema( | ||||||
| @@ -197,8 +254,9 @@ class HassioAddonConfig extends LitElement { | |||||||
|   protected firstUpdated(changedProps) { |   protected firstUpdated(changedProps) { | ||||||
|     super.firstUpdated(changedProps); |     super.firstUpdated(changedProps); | ||||||
|     this._canShowSchema = !this.addon.schema!.find( |     this._canShowSchema = !this.addon.schema!.find( | ||||||
|  |       (entry) => | ||||||
|         // @ts-ignore |         // @ts-ignore | ||||||
|       (entry) => !SUPPORTED_UI_TYPES.includes(entry.type) || entry.multiple |         !SUPPORTED_UI_TYPES.includes(entry.type) | ||||||
|     ); |     ); | ||||||
|     this._yamlMode = !this._canShowSchema; |     this._yamlMode = !this._canShowSchema; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import { PaperInputElement } from "@polymer/paper-input/paper-input"; |  | ||||||
| import { | import { | ||||||
|   css, |   css, | ||||||
|   CSSResultGroup, |   CSSResultGroup, | ||||||
| @@ -8,10 +7,13 @@ import { | |||||||
|   TemplateResult, |   TemplateResult, | ||||||
| } from "lit"; | } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
|  | import memoizeOne from "memoize-one"; | ||||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||||
| import "../../../../src/components/buttons/ha-progress-button"; | import "../../../../src/components/buttons/ha-progress-button"; | ||||||
| import "../../../../src/components/ha-alert"; | import "../../../../src/components/ha-alert"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-form/ha-form"; | ||||||
|  | import type { HaFormSchema } from "../../../../src/components/ha-form/types"; | ||||||
| import { | import { | ||||||
|   HassioAddonDetails, |   HassioAddonDetails, | ||||||
|   HassioAddonSetOptionParams, |   HassioAddonSetOptionParams, | ||||||
| @@ -24,16 +26,6 @@ import { HomeAssistant } from "../../../../src/types"; | |||||||
| import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart"; | import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart"; | ||||||
| import { hassioStyle } from "../../resources/hassio-style"; | import { hassioStyle } from "../../resources/hassio-style"; | ||||||
|  |  | ||||||
| interface NetworkItem { |  | ||||||
|   description: string; |  | ||||||
|   container: string; |  | ||||||
|   host: number | null; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| interface NetworkItemInput extends PaperInputElement { |  | ||||||
|   container: string; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| @customElement("hassio-addon-network") | @customElement("hassio-addon-network") | ||||||
| class HassioAddonNetwork extends LitElement { | class HassioAddonNetwork extends LitElement { | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |   @property({ attribute: false }) public hass!: HomeAssistant; | ||||||
| @@ -42,9 +34,13 @@ class HassioAddonNetwork extends LitElement { | |||||||
|  |  | ||||||
|   @property({ attribute: false }) public addon!: HassioAddonDetails; |   @property({ attribute: false }) public addon!: HassioAddonDetails; | ||||||
|  |  | ||||||
|  |   @state() private _showOptional = false; | ||||||
|  |  | ||||||
|  |   @state() private _configHasChanged = false; | ||||||
|  |  | ||||||
|   @state() private _error?: string; |   @state() private _error?: string; | ||||||
|  |  | ||||||
|   @state() private _config?: NetworkItem[]; |   @state() private _config?: Record<string, any>; | ||||||
|  |  | ||||||
|   public connectedCallback(): void { |   public connectedCallback(): void { | ||||||
|     super.connectedCallback(); |     super.connectedCallback(); | ||||||
| @@ -56,59 +52,61 @@ class HassioAddonNetwork extends LitElement { | |||||||
|       return html``; |       return html``; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     const hasHiddenOptions = Object.keys(this._config).find( | ||||||
|  |       (entry) => this._config![entry] === null | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-card |       <ha-card | ||||||
|  |         outlined | ||||||
|         .header=${this.supervisor.localize( |         .header=${this.supervisor.localize( | ||||||
|           "addon.configuration.network.header" |           "addon.configuration.network.header" | ||||||
|         )} |         )} | ||||||
|       > |       > | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|  |           <p> | ||||||
|  |             ${this.supervisor.localize( | ||||||
|  |               "addon.configuration.network.introduction" | ||||||
|  |             )} | ||||||
|  |           </p> | ||||||
|           ${this._error |           ${this._error | ||||||
|             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` |             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||||
|             : ""} |             : ""} | ||||||
|  |  | ||||||
|           <table> |           <ha-form | ||||||
|             <tbody> |             .data=${this._config} | ||||||
|               <tr> |  | ||||||
|                 <th> |  | ||||||
|                   ${this.supervisor.localize( |  | ||||||
|                     "addon.configuration.network.container" |  | ||||||
|                   )} |  | ||||||
|                 </th> |  | ||||||
|                 <th> |  | ||||||
|                   ${this.supervisor.localize( |  | ||||||
|                     "addon.configuration.network.host" |  | ||||||
|                   )} |  | ||||||
|                 </th> |  | ||||||
|                 <th>${this.supervisor.localize("common.description")}</th> |  | ||||||
|               </tr> |  | ||||||
|               ${this._config!.map( |  | ||||||
|                 (item) => html` |  | ||||||
|                   <tr> |  | ||||||
|                     <td>${item.container}</td> |  | ||||||
|                     <td> |  | ||||||
|                       <paper-input |  | ||||||
|             @value-changed=${this._configChanged} |             @value-changed=${this._configChanged} | ||||||
|                         placeholder=${this.supervisor.localize( |             .computeLabel=${this._computeLabel} | ||||||
|                           "addon.configuration.network.disabled" |             .computeHelper=${this._computeHelper} | ||||||
|  |             .schema=${this._createSchema( | ||||||
|  |               this._config, | ||||||
|  |               this._showOptional, | ||||||
|  |               this.hass.userData?.showAdvanced || false | ||||||
|             )} |             )} | ||||||
|                         .value=${item.host ? String(item.host) : ""} |           ></ha-form> | ||||||
|                         .container=${item.container} |  | ||||||
|                         no-label-float |  | ||||||
|                       ></paper-input> |  | ||||||
|                     </td> |  | ||||||
|                     <td>${this._computeDescription(item)}</td> |  | ||||||
|                   </tr> |  | ||||||
|                 ` |  | ||||||
|               )} |  | ||||||
|             </tbody> |  | ||||||
|           </table> |  | ||||||
|         </div> |         </div> | ||||||
|  |         ${hasHiddenOptions | ||||||
|  |           ? html`<ha-formfield | ||||||
|  |               class="show-optional" | ||||||
|  |               .label=${this.supervisor.localize( | ||||||
|  |                 "addon.configuration.network.show_disabled" | ||||||
|  |               )} | ||||||
|  |             > | ||||||
|  |               <ha-switch | ||||||
|  |                 @change=${this._toggleOptional} | ||||||
|  |                 .checked=${this._showOptional} | ||||||
|  |               > | ||||||
|  |               </ha-switch> | ||||||
|  |             </ha-formfield>` | ||||||
|  |           : ""} | ||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           <ha-progress-button class="warning" @click=${this._resetTapped}> |           <ha-progress-button class="warning" @click=${this._resetTapped}> | ||||||
|             ${this.supervisor.localize("common.reset_defaults")} |             ${this.supervisor.localize("common.reset_defaults")} | ||||||
|           </ha-progress-button> |           </ha-progress-button> | ||||||
|           <ha-progress-button @click=${this._saveTapped}> |           <ha-progress-button | ||||||
|  |             @click=${this._saveTapped} | ||||||
|  |             .disabled=${!this._configHasChanged} | ||||||
|  |           > | ||||||
|             ${this.supervisor.localize("common.save")} |             ${this.supervisor.localize("common.save")} | ||||||
|           </ha-progress-button> |           </ha-progress-button> | ||||||
|         </div> |         </div> | ||||||
| @@ -123,50 +121,60 @@ class HassioAddonNetwork extends LitElement { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _computeDescription = (item: NetworkItem): string => |   private _createSchema = memoizeOne( | ||||||
|     this.addon.translations[this.hass.language]?.network?.[item.container] |     ( | ||||||
|       ?.description || |       config: Record<string, number>, | ||||||
|     this.addon.translations.en?.network?.[item.container]?.description || |       showOptional: boolean, | ||||||
|     item.description; |       advanced: boolean | ||||||
|  |     ): HaFormSchema[] => | ||||||
|  |       (showOptional | ||||||
|  |         ? Object.keys(config) | ||||||
|  |         : Object.keys(config).filter((entry) => config[entry] !== null) | ||||||
|  |       ).map((entry) => ({ | ||||||
|  |         name: entry, | ||||||
|  |         selector: { | ||||||
|  |           number: { | ||||||
|  |             mode: "box", | ||||||
|  |             min: 0, | ||||||
|  |             max: 65535, | ||||||
|  |             unit_of_measurement: advanced ? entry : undefined, | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       })) | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   private _computeLabel = (_: HaFormSchema): string => ""; | ||||||
|  |  | ||||||
|  |   private _computeHelper = (item: HaFormSchema): string => | ||||||
|  |     this.addon.translations[this.hass.language]?.network?.[item.name] || | ||||||
|  |     this.addon.translations.en?.network?.[item.name] || | ||||||
|  |     this.addon.network_description?.[item.name] || | ||||||
|  |     item.name; | ||||||
|  |  | ||||||
|   private _setNetworkConfig(): void { |   private _setNetworkConfig(): void { | ||||||
|     const network = this.addon.network || {}; |     this._config = this.addon.network || {}; | ||||||
|     const description = this.addon.network_description || {}; |  | ||||||
|     const items: NetworkItem[] = Object.keys(network).map((key) => ({ |  | ||||||
|       container: key, |  | ||||||
|       host: network[key], |  | ||||||
|       description: description[key], |  | ||||||
|     })); |  | ||||||
|     this._config = items.sort((a, b) => (a.container > b.container ? 1 : -1)); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _configChanged(ev: Event): Promise<void> { |   private async _configChanged(ev: CustomEvent): Promise<void> { | ||||||
|     const target = ev.target as NetworkItemInput; |     this._configHasChanged = true; | ||||||
|     this._config!.forEach((item) => { |     this._config! = ev.detail.value; | ||||||
|       if ( |  | ||||||
|         item.container === target.container && |  | ||||||
|         item.host !== parseInt(String(target.value), 10) |  | ||||||
|       ) { |  | ||||||
|         item.host = target.value ? parseInt(String(target.value), 10) : null; |  | ||||||
|       } |  | ||||||
|     }); |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _resetTapped(ev: CustomEvent): Promise<void> { |   private async _resetTapped(ev: CustomEvent): Promise<void> { | ||||||
|     const button = ev.currentTarget as any; |     const button = ev.currentTarget as any; | ||||||
|     button.progress = true; |  | ||||||
|  |  | ||||||
|     const data: HassioAddonSetOptionParams = { |     const data: HassioAddonSetOptionParams = { | ||||||
|       network: null, |       network: null, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     try { |     try { | ||||||
|       await setHassioAddonOption(this.hass, this.addon.slug, data); |       await setHassioAddonOption(this.hass, this.addon.slug, data); | ||||||
|  |       this._configHasChanged = false; | ||||||
|       const eventdata = { |       const eventdata = { | ||||||
|         success: true, |         success: true, | ||||||
|         response: undefined, |         response: undefined, | ||||||
|         path: "option", |         path: "option", | ||||||
|       }; |       }; | ||||||
|  |       button.actionSuccess(); | ||||||
|       fireEvent(this, "hass-api-called", eventdata); |       fireEvent(this, "hass-api-called", eventdata); | ||||||
|       if (this.addon?.state === "started") { |       if (this.addon?.state === "started") { | ||||||
|         await suggestAddonRestart(this, this.hass, this.supervisor, this.addon); |         await suggestAddonRestart(this, this.hass, this.supervisor, this.addon); | ||||||
| @@ -177,19 +185,21 @@ class HassioAddonNetwork extends LitElement { | |||||||
|         "error", |         "error", | ||||||
|         extractApiErrorMessage(err) |         extractApiErrorMessage(err) | ||||||
|       ); |       ); | ||||||
|  |       button.actionError(); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|     button.progress = false; |   private _toggleOptional() { | ||||||
|  |     this._showOptional = !this._showOptional; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private async _saveTapped(ev: CustomEvent): Promise<void> { |   private async _saveTapped(ev: CustomEvent): Promise<void> { | ||||||
|     const button = ev.currentTarget as any; |     const button = ev.currentTarget as any; | ||||||
|     button.progress = true; |  | ||||||
|  |  | ||||||
|     this._error = undefined; |     this._error = undefined; | ||||||
|     const networkconfiguration = {}; |     const networkconfiguration = {}; | ||||||
|     this._config!.forEach((item) => { |     Object.entries(this._config!).forEach(([key, value]) => { | ||||||
|       networkconfiguration[item.container] = parseInt(String(item.host), 10); |       networkconfiguration[key] = value ?? null; | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     const data: HassioAddonSetOptionParams = { |     const data: HassioAddonSetOptionParams = { | ||||||
| @@ -198,11 +208,13 @@ class HassioAddonNetwork extends LitElement { | |||||||
|  |  | ||||||
|     try { |     try { | ||||||
|       await setHassioAddonOption(this.hass, this.addon.slug, data); |       await setHassioAddonOption(this.hass, this.addon.slug, data); | ||||||
|  |       this._configHasChanged = false; | ||||||
|       const eventdata = { |       const eventdata = { | ||||||
|         success: true, |         success: true, | ||||||
|         response: undefined, |         response: undefined, | ||||||
|         path: "option", |         path: "option", | ||||||
|       }; |       }; | ||||||
|  |       button.actionSuccess(); | ||||||
|       fireEvent(this, "hass-api-called", eventdata); |       fireEvent(this, "hass-api-called", eventdata); | ||||||
|       if (this.addon?.state === "started") { |       if (this.addon?.state === "started") { | ||||||
|         await suggestAddonRestart(this, this.hass, this.supervisor, this.addon); |         await suggestAddonRestart(this, this.hass, this.supervisor, this.addon); | ||||||
| @@ -213,8 +225,8 @@ class HassioAddonNetwork extends LitElement { | |||||||
|         "error", |         "error", | ||||||
|         extractApiErrorMessage(err) |         extractApiErrorMessage(err) | ||||||
|       ); |       ); | ||||||
|  |       button.actionError(); | ||||||
|     } |     } | ||||||
|     button.progress = false; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static get styles(): CSSResultGroup { |   static get styles(): CSSResultGroup { | ||||||
| @@ -232,6 +244,9 @@ class HassioAddonNetwork extends LitElement { | |||||||
|           display: flex; |           display: flex; | ||||||
|           justify-content: space-between; |           justify-content: space-between; | ||||||
|         } |         } | ||||||
|  |         .show-optional { | ||||||
|  |           padding: 16px; | ||||||
|  |         } | ||||||
|       `, |       `, | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -38,7 +38,7 @@ class HassioAddonDocumentationDashboard extends LitElement { | |||||||
|     } |     } | ||||||
|     return html` |     return html` | ||||||
|       <div class="content"> |       <div class="content"> | ||||||
|         <ha-card> |         <ha-card outlined> | ||||||
|           ${this._error |           ${this._error | ||||||
|             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` |             ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||||
|             : ""} |             : ""} | ||||||
|   | |||||||
| @@ -17,7 +17,12 @@ import { | |||||||
|   HassioAddonDetails, |   HassioAddonDetails, | ||||||
| } from "../../../src/data/hassio/addon"; | } from "../../../src/data/hassio/addon"; | ||||||
| import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | ||||||
|  | import { | ||||||
|  |   fetchHassioSupervisorInfo, | ||||||
|  |   setSupervisorOption, | ||||||
|  | } from "../../../src/data/hassio/supervisor"; | ||||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||||
|  | import { showConfirmationDialog } from "../../../src/dialogs/generic/show-dialog-box"; | ||||||
| import "../../../src/layouts/hass-error-screen"; | import "../../../src/layouts/hass-error-screen"; | ||||||
| import "../../../src/layouts/hass-loading-screen"; | import "../../../src/layouts/hass-loading-screen"; | ||||||
| import "../../../src/layouts/hass-tabs-subpage"; | import "../../../src/layouts/hass-tabs-subpage"; | ||||||
| @@ -166,6 +171,44 @@ class HassioAddonDashboard extends LitElement { | |||||||
|   protected async firstUpdated(): Promise<void> { |   protected async firstUpdated(): Promise<void> { | ||||||
|     if (this.route.path === "") { |     if (this.route.path === "") { | ||||||
|       const requestedAddon = extractSearchParam("addon"); |       const requestedAddon = extractSearchParam("addon"); | ||||||
|  |       const requestedAddonRepository = extractSearchParam("repository_url"); | ||||||
|  |       if (requestedAddonRepository) { | ||||||
|  |         const supervisorInfo = await fetchHassioSupervisorInfo(this.hass); | ||||||
|  |         if ( | ||||||
|  |           !supervisorInfo.addons_repositories.find( | ||||||
|  |             (repo) => repo === requestedAddonRepository | ||||||
|  |           ) | ||||||
|  |         ) { | ||||||
|  |           if ( | ||||||
|  |             !(await showConfirmationDialog(this, { | ||||||
|  |               title: this.supervisor.localize("my.add_addon_repository_title"), | ||||||
|  |               text: this.supervisor.localize( | ||||||
|  |                 "my.add_addon_repository_description", | ||||||
|  |                 { addon: requestedAddon, repository: requestedAddonRepository } | ||||||
|  |               ), | ||||||
|  |               confirmText: this.supervisor.localize("common.add"), | ||||||
|  |               dismissText: this.supervisor.localize("common.cancel"), | ||||||
|  |             })) | ||||||
|  |           ) { | ||||||
|  |             this._error = this.supervisor.localize( | ||||||
|  |               "my.error_repository_not_found" | ||||||
|  |             ); | ||||||
|  |             return; | ||||||
|  |           } | ||||||
|  |  | ||||||
|  |           try { | ||||||
|  |             await setSupervisorOption(this.hass, { | ||||||
|  |               addons_repositories: [ | ||||||
|  |                 ...supervisorInfo.addons_repositories, | ||||||
|  |                 requestedAddonRepository, | ||||||
|  |               ], | ||||||
|  |             }); | ||||||
|  |           } catch (err: any) { | ||||||
|  |             this._error = extractApiErrorMessage(err); | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|       if (requestedAddon) { |       if (requestedAddon) { | ||||||
|         const addonsInfo = await fetchHassioAddonsInfo(this.hass); |         const addonsInfo = await fetchHassioAddonsInfo(this.hass); | ||||||
|         const validAddon = addonsInfo.addons.some( |         const validAddon = addonsInfo.addons.some( | ||||||
|   | |||||||
| @@ -166,7 +166,7 @@ class HassioAddonInfo extends LitElement { | |||||||
|           ` |           ` | ||||||
|         : ""} |         : ""} | ||||||
|  |  | ||||||
|       <ha-card> |       <ha-card outlined> | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|           <div class="addon-header"> |           <div class="addon-header"> | ||||||
|             ${!this.narrow ? this.addon.name : ""} |             ${!this.narrow ? this.addon.name : ""} | ||||||
| @@ -649,7 +649,7 @@ class HassioAddonInfo extends LitElement { | |||||||
|  |  | ||||||
|       ${this.addon.long_description |       ${this.addon.long_description | ||||||
|         ? html` |         ? html` | ||||||
|             <ha-card> |             <ha-card outlined> | ||||||
|               <div class="card-content"> |               <div class="card-content"> | ||||||
|                 <ha-markdown |                 <ha-markdown | ||||||
|                   .content=${this.addon.long_description} |                   .content=${this.addon.long_description} | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import "@material/mwc-button"; | |||||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import "../../../../src/components/ha-alert"; | import "../../../../src/components/ha-alert"; | ||||||
|  | import "../../../../src/components/ha-ansi-to-html"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
| import { | import { | ||||||
|   fetchHassioAddonLogs, |   fetchHassioAddonLogs, | ||||||
| @@ -11,7 +12,6 @@ import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | |||||||
| import { Supervisor } from "../../../../src/data/supervisor/supervisor"; | import { Supervisor } from "../../../../src/data/supervisor/supervisor"; | ||||||
| import { haStyle } from "../../../../src/resources/styles"; | import { haStyle } from "../../../../src/resources/styles"; | ||||||
| import { HomeAssistant } from "../../../../src/types"; | import { HomeAssistant } from "../../../../src/types"; | ||||||
| import "../../components/hassio-ansi-to-html"; |  | ||||||
| import { hassioStyle } from "../../resources/hassio-style"; | import { hassioStyle } from "../../resources/hassio-style"; | ||||||
|  |  | ||||||
| @customElement("hassio-addon-logs") | @customElement("hassio-addon-logs") | ||||||
| @@ -34,15 +34,15 @@ class HassioAddonLogs extends LitElement { | |||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     return html` |     return html` | ||||||
|       <h1>${this.addon.name}</h1> |       <h1>${this.addon.name}</h1> | ||||||
|       <ha-card> |       <ha-card outlined> | ||||||
|         ${this._error |         ${this._error | ||||||
|           ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` |           ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||||
|           : ""} |           : ""} | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|           ${this._content |           ${this._content | ||||||
|             ? html`<hassio-ansi-to-html |             ? html`<ha-ansi-to-html | ||||||
|                 .content=${this._content} |                 .content=${this._content} | ||||||
|               ></hassio-ansi-to-html>` |               ></ha-ansi-to-html>` | ||||||
|             : ""} |             : ""} | ||||||
|         </div> |         </div> | ||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| import "@material/mwc-button"; | import "@material/mwc-button"; | ||||||
| import { ActionDetail } from "@material/mwc-list"; | import { ActionDetail } from "@material/mwc-list"; | ||||||
| import "@material/mwc-list/mwc-list-item"; | import "@material/mwc-list/mwc-list-item"; | ||||||
| import { mdiDelete, mdiDotsVertical, mdiPlus } from "@mdi/js"; | import { mdiBackupRestore, mdiDelete, mdiDotsVertical, mdiPlus } from "@mdi/js"; | ||||||
| import { | import { | ||||||
|   css, |   css, | ||||||
|   CSSResultGroup, |   CSSResultGroup, | ||||||
| @@ -166,7 +166,15 @@ export class HassioBackups extends LitElement { | |||||||
|     } |     } | ||||||
|     return html` |     return html` | ||||||
|       <hass-tabs-subpage-data-table |       <hass-tabs-subpage-data-table | ||||||
|         .tabs=${supervisorTabs(this.hass)} |         .tabs=${atLeastVersion(this.hass.config.version, 2022, 5) | ||||||
|  |           ? [ | ||||||
|  |               { | ||||||
|  |                 translationKey: "panel.backups", | ||||||
|  |                 path: `/hassio/backups`, | ||||||
|  |                 iconPath: mdiBackupRestore, | ||||||
|  |               }, | ||||||
|  |             ] | ||||||
|  |           : supervisorTabs(this.hass)} | ||||||
|         .hass=${this.hass} |         .hass=${this.hass} | ||||||
|         .localizeFunc=${this.supervisor.localize} |         .localizeFunc=${this.supervisor.localize} | ||||||
|         .searchLabel=${this.supervisor.localize("search")} |         .searchLabel=${this.supervisor.localize("search")} | ||||||
| @@ -182,7 +190,9 @@ export class HassioBackups extends LitElement { | |||||||
|         selectable |         selectable | ||||||
|         hasFab |         hasFab | ||||||
|         .mainPage=${!atLeastVersion(this.hass.config.version, 2021, 12)} |         .mainPage=${!atLeastVersion(this.hass.config.version, 2021, 12)} | ||||||
|         back-path="/config" |         back-path=${atLeastVersion(this.hass.config.version, 2022, 5) | ||||||
|  |           ? "/config/system" | ||||||
|  |           : "/config"} | ||||||
|         supervisor |         supervisor | ||||||
|       > |       > | ||||||
|         <ha-button-menu |         <ha-button-menu | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| import { mdiFolderUpload } from "@mdi/js"; | import { mdiFolderUpload } from "@mdi/js"; | ||||||
| import "@polymer/paper-input/paper-input-container"; |  | ||||||
| import { html, LitElement, TemplateResult } from "lit"; | import { html, LitElement, TemplateResult } from "lit"; | ||||||
| import { customElement, state } from "lit/decorators"; | import { customElement, state } from "lit/decorators"; | ||||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||||
|   | |||||||
| @@ -32,13 +32,6 @@ interface AddonCheckboxItem extends CheckboxItem { | |||||||
|  |  | ||||||
| const _computeFolders = (folders): CheckboxItem[] => { | const _computeFolders = (folders): CheckboxItem[] => { | ||||||
|   const list: CheckboxItem[] = []; |   const list: CheckboxItem[] = []; | ||||||
|   if (folders.includes("homeassistant")) { |  | ||||||
|     list.push({ |  | ||||||
|       slug: "homeassistant", |  | ||||||
|       name: "Home Assistant configuration", |  | ||||||
|       checked: false, |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
|   if (folders.includes("ssl")) { |   if (folders.includes("ssl")) { | ||||||
|     list.push({ slug: "ssl", name: "SSL", checked: false }); |     list.push({ slug: "ssl", name: "SSL", checked: false }); | ||||||
|   } |   } | ||||||
| @@ -100,7 +93,7 @@ export class SupervisorBackupContent extends LitElement { | |||||||
|       this.folders = _computeFolders( |       this.folders = _computeFolders( | ||||||
|         this.backup |         this.backup | ||||||
|           ? this.backup.folders |           ? this.backup.folders | ||||||
|           : ["homeassistant", "ssl", "share", "media", "addons/local"] |           : ["ssl", "share", "media", "addons/local"] | ||||||
|       ); |       ); | ||||||
|       this.addons = _computeAddons( |       this.addons = _computeAddons( | ||||||
|         this.backup ? this.backup.addons : this.supervisor?.supervisor.addons |         this.backup ? this.backup.addons : this.supervisor?.supervisor.addons | ||||||
| @@ -187,7 +180,7 @@ export class SupervisorBackupContent extends LitElement { | |||||||
|             > |             > | ||||||
|               <ha-checkbox |               <ha-checkbox | ||||||
|                 .checked=${this.homeAssistant} |                 .checked=${this.homeAssistant} | ||||||
|                 @click=${this.toggleHomeAssistant} |                 @change=${this.toggleHomeAssistant} | ||||||
|               > |               > | ||||||
|               </ha-checkbox> |               </ha-checkbox> | ||||||
|             </ha-formfield> |             </ha-formfield> | ||||||
|   | |||||||
| @@ -26,7 +26,7 @@ class HassioAddons extends LitElement { | |||||||
|         <div class="card-group"> |         <div class="card-group"> | ||||||
|           ${!this.supervisor.supervisor.addons?.length |           ${!this.supervisor.supervisor.addons?.length | ||||||
|             ? html` |             ? html` | ||||||
|                 <ha-card> |                 <ha-card outlined> | ||||||
|                   <div class="card-content"> |                   <div class="card-content"> | ||||||
|                     <button class="link" @click=${this._openStore}> |                     <button class="link" @click=${this._openStore}> | ||||||
|                       ${this.supervisor.localize("dashboard.no_addons")} |                       ${this.supervisor.localize("dashboard.no_addons")} | ||||||
| @@ -38,7 +38,11 @@ class HassioAddons extends LitElement { | |||||||
|                 .sort((a, b) => caseInsensitiveStringCompare(a.name, b.name)) |                 .sort((a, b) => caseInsensitiveStringCompare(a.name, b.name)) | ||||||
|                 .map( |                 .map( | ||||||
|                   (addon) => html` |                   (addon) => html` | ||||||
|                     <ha-card .addon=${addon} @click=${this._addonTapped}> |                     <ha-card | ||||||
|  |                       outlined | ||||||
|  |                       .addon=${addon} | ||||||
|  |                       @click=${this._addonTapped} | ||||||
|  |                     > | ||||||
|                       <div class="card-content"> |                       <div class="card-content"> | ||||||
|                         <hassio-card-content |                         <hassio-card-content | ||||||
|                           .hass=${this.hass} |                           .hass=${this.hass} | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ import { HomeAssistant, Route } from "../../../src/types"; | |||||||
| import { supervisorTabs } from "../hassio-tabs"; | import { supervisorTabs } from "../hassio-tabs"; | ||||||
| import "./hassio-addons"; | import "./hassio-addons"; | ||||||
| import "./hassio-update"; | import "./hassio-update"; | ||||||
|  | import "../../../src/layouts/hass-subpage"; | ||||||
|  |  | ||||||
| @customElement("hassio-dashboard") | @customElement("hassio-dashboard") | ||||||
| class HassioDashboard extends LitElement { | class HassioDashboard extends LitElement { | ||||||
| @@ -22,6 +23,31 @@ class HassioDashboard extends LitElement { | |||||||
|   @property({ attribute: false }) public route!: Route; |   @property({ attribute: false }) public route!: Route; | ||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|  |     if (atLeastVersion(this.hass.config.version, 2022, 5)) { | ||||||
|  |       return html`<hass-subpage | ||||||
|  |         .hass=${this.hass} | ||||||
|  |         .narrow=${this.narrow} | ||||||
|  |         .route=${this.route} | ||||||
|  |         .header=${this.supervisor.localize("panel.addons")} | ||||||
|  |       > | ||||||
|  |         <hassio-addons | ||||||
|  |           .hass=${this.hass} | ||||||
|  |           .supervisor=${this.supervisor} | ||||||
|  |         ></hassio-addons> | ||||||
|  |         <a href="/hassio/store"> | ||||||
|  |           <ha-fab | ||||||
|  |             .label=${this.supervisor.localize("panel.store")} | ||||||
|  |             extended | ||||||
|  |             class="non-tabs" | ||||||
|  |           > | ||||||
|  |             <ha-svg-icon | ||||||
|  |               slot="icon" | ||||||
|  |               .path=${mdiStorePlus} | ||||||
|  |             ></ha-svg-icon> </ha-fab | ||||||
|  |         ></a> | ||||||
|  |       </hass-subpage>`; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <hass-tabs-subpage |       <hass-tabs-subpage | ||||||
|         .hass=${this.hass} |         .hass=${this.hass} | ||||||
| @@ -74,6 +100,12 @@ class HassioDashboard extends LitElement { | |||||||
|         .content { |         .content { | ||||||
|           margin: 0 auto; |           margin: 0 auto; | ||||||
|         } |         } | ||||||
|  |         ha-fab.non-tabs { | ||||||
|  |           position: fixed; | ||||||
|  |           right: calc(16px + env(safe-area-inset-right)); | ||||||
|  |           bottom: calc(16px + env(safe-area-inset-bottom)); | ||||||
|  |           z-index: 1; | ||||||
|  |         } | ||||||
|       `, |       `, | ||||||
|     ]; |     ]; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -85,7 +85,7 @@ export class HassioUpdate extends LitElement { | |||||||
|       return html``; |       return html``; | ||||||
|     } |     } | ||||||
|     return html` |     return html` | ||||||
|       <ha-card> |       <ha-card outlined> | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|           <div class="icon"> |           <div class="icon"> | ||||||
|             <ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon> |             <ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon> | ||||||
|   | |||||||
| @@ -1,11 +1,11 @@ | |||||||
| import "@material/mwc-list/mwc-list-item"; | import "@material/mwc-list/mwc-list-item"; | ||||||
| import "@material/mwc-select"; |  | ||||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||||
| import "../../../../src/components/ha-circular-progress"; | import "../../../../src/components/ha-circular-progress"; | ||||||
| import "../../../../src/components/ha-markdown"; | import "../../../../src/components/ha-markdown"; | ||||||
|  | import "../../../../src/components/ha-select"; | ||||||
| import { | import { | ||||||
|   extractApiErrorMessage, |   extractApiErrorMessage, | ||||||
|   ignoreSupervisorError, |   ignoreSupervisorError, | ||||||
| @@ -89,7 +89,7 @@ class HassioDatadiskDialog extends LitElement { | |||||||
|                     )} |                     )} | ||||||
|                     <br /><br /> |                     <br /><br /> | ||||||
|  |  | ||||||
|                     <mwc-select |                     <ha-select | ||||||
|                       .label=${this.dialogParams.supervisor.localize( |                       .label=${this.dialogParams.supervisor.localize( | ||||||
|                         "dialog.datadisk_move.select_device" |                         "dialog.datadisk_move.select_device" | ||||||
|                       )} |                       )} | ||||||
| @@ -102,7 +102,7 @@ class HassioDatadiskDialog extends LitElement { | |||||||
|                             >${device}</mwc-list-item |                             >${device}</mwc-list-item | ||||||
|                           >` |                           >` | ||||||
|                       )} |                       )} | ||||||
|                     </mwc-select> |                     </ha-select> | ||||||
|                   ` |                   ` | ||||||
|                 : this.devices === undefined |                 : this.devices === undefined | ||||||
|                 ? this.dialogParams.supervisor.localize( |                 ? this.dialogParams.supervisor.localize( | ||||||
| @@ -161,7 +161,7 @@ class HassioDatadiskDialog extends LitElement { | |||||||
|       haStyle, |       haStyle, | ||||||
|       haStyleDialog, |       haStyleDialog, | ||||||
|       css` |       css` | ||||||
|         mwc-select { |         ha-select { | ||||||
|           width: 100%; |           width: 100%; | ||||||
|         } |         } | ||||||
|         ha-circular-progress { |         ha-circular-progress { | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | |||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||||
| import "../../../../src/common/search/search-input"; | import "../../../../src/components/search-input"; | ||||||
| import { stringCompare } from "../../../../src/common/string/compare"; | import { stringCompare } from "../../../../src/common/string/compare"; | ||||||
| import "../../../../src/components/ha-dialog"; | import "../../../../src/components/ha-dialog"; | ||||||
| import "../../../../src/components/ha-expansion-panel"; | import "../../../../src/components/ha-expansion-panel"; | ||||||
| @@ -80,8 +80,6 @@ class HassioHardwareDialog extends LitElement { | |||||||
|           ></ha-icon-button> |           ></ha-icon-button> | ||||||
|           <search-input |           <search-input | ||||||
|             .hass=${this.hass} |             .hass=${this.hass} | ||||||
|             dialogInitialFocus |  | ||||||
|             no-label-float |  | ||||||
|             .filter=${this._filter} |             .filter=${this._filter} | ||||||
|             @value-changed=${this._handleSearchChange} |             @value-changed=${this._handleSearchChange} | ||||||
|             .label=${this._dialogParams.supervisor.localize( |             .label=${this._dialogParams.supervisor.localize( | ||||||
|   | |||||||
| @@ -106,6 +106,9 @@ class HassioRepositoriesDialog extends LitElement { | |||||||
|                     </paper-item-body> |                     </paper-item-body> | ||||||
|                     <div class="delete"> |                     <div class="delete"> | ||||||
|                       <ha-icon-button |                       <ha-icon-button | ||||||
|  |                         .label=${this._dialogParams!.supervisor.localize( | ||||||
|  |                           "dialog.repositories.remove" | ||||||
|  |                         )} | ||||||
|                         .disabled=${usedRepositories.includes(repo.slug)} |                         .disabled=${usedRepositories.includes(repo.slug)} | ||||||
|                         .slug=${repo.slug} |                         .slug=${repo.slug} | ||||||
|                         .path=${usedRepositories.includes(repo.slug) |                         .path=${usedRepositories.includes(repo.slug) | ||||||
|   | |||||||
| @@ -1,9 +1,12 @@ | |||||||
| // Compat needs to be first import | // Compat needs to be first import | ||||||
| import "../../src/resources/compatibility"; | import "../../src/resources/compatibility"; | ||||||
|  | import { setCancelSyntheticClickEvents } from "@polymer/polymer/lib/utils/settings"; | ||||||
| import "../../src/resources/roboto"; | import "../../src/resources/roboto"; | ||||||
| import "../../src/resources/safari-14-attachshadow-patch"; | import "../../src/resources/safari-14-attachshadow-patch"; | ||||||
| import "./hassio-main"; | import "./hassio-main"; | ||||||
|  |  | ||||||
|  | setCancelSyntheticClickEvents(false); | ||||||
|  |  | ||||||
| const styleEl = document.createElement("style"); | const styleEl = document.createElement("style"); | ||||||
| styleEl.innerHTML = ` | styleEl.innerHTML = ` | ||||||
| body { | body { | ||||||
|   | |||||||
| @@ -3,8 +3,8 @@ import { customElement, property } from "lit/decorators"; | |||||||
| import { atLeastVersion } from "../../src/common/config/version"; | import { atLeastVersion } from "../../src/common/config/version"; | ||||||
| import { applyThemesOnElement } from "../../src/common/dom/apply_themes_on_element"; | import { applyThemesOnElement } from "../../src/common/dom/apply_themes_on_element"; | ||||||
| import { fireEvent } from "../../src/common/dom/fire_event"; | import { fireEvent } from "../../src/common/dom/fire_event"; | ||||||
| import { isNavigationClick } from "../../src/common/dom/is-navigation-click"; |  | ||||||
| import { mainWindow } from "../../src/common/dom/get_main_window"; | import { mainWindow } from "../../src/common/dom/get_main_window"; | ||||||
|  | import { isNavigationClick } from "../../src/common/dom/is-navigation-click"; | ||||||
| import { navigate } from "../../src/common/navigate"; | import { navigate } from "../../src/common/navigate"; | ||||||
| import { HassioPanelInfo } from "../../src/data/hassio/supervisor"; | import { HassioPanelInfo } from "../../src/data/hassio/supervisor"; | ||||||
| import { Supervisor } from "../../src/data/supervisor/supervisor"; | import { Supervisor } from "../../src/data/supervisor/supervisor"; | ||||||
| @@ -73,6 +73,18 @@ export class HassioMain extends SupervisorBaseElement { | |||||||
|       }); |       }); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     // Forward keydown events to the main window for quickbar access | ||||||
|  |     document.body.addEventListener("keydown", (ev: KeyboardEvent) => { | ||||||
|  |       if (ev.altKey || ev.ctrlKey || ev.shiftKey || ev.metaKey) { | ||||||
|  |         // Ignore if modifier keys are pressed | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |       // @ts-ignore | ||||||
|  |       fireEvent(mainWindow, "hass-quick-bar-trigger", ev, { | ||||||
|  |         bubbles: false, | ||||||
|  |       }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|     makeDialogManager(this, this.shadowRoot!); |     makeDialogManager(this, this.shadowRoot!); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -121,7 +133,8 @@ export class HassioMain extends SupervisorBaseElement { | |||||||
|       this.parentElement, |       this.parentElement, | ||||||
|       this.hass.themes, |       this.hass.themes, | ||||||
|       themeName, |       themeName, | ||||||
|       themeSettings |       themeSettings, | ||||||
|  |       true | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ import { | |||||||
| } from "../../src/panels/my/ha-panel-my"; | } from "../../src/panels/my/ha-panel-my"; | ||||||
| import { HomeAssistant, Route } from "../../src/types"; | import { HomeAssistant, Route } from "../../src/types"; | ||||||
|  |  | ||||||
| const REDIRECTS: Redirects = { | export const REDIRECTS: Redirects = { | ||||||
|   supervisor: { |   supervisor: { | ||||||
|     redirect: "/hassio/dashboard", |     redirect: "/hassio/dashboard", | ||||||
|   }, |   }, | ||||||
| @@ -42,6 +42,9 @@ const REDIRECTS: Redirects = { | |||||||
|     params: { |     params: { | ||||||
|       addon: "string", |       addon: "string", | ||||||
|     }, |     }, | ||||||
|  |     optional_params: { | ||||||
|  |       repository_url: "url", | ||||||
|  |     }, | ||||||
|   }, |   }, | ||||||
|   supervisor_ingress: { |   supervisor_ingress: { | ||||||
|     redirect: "/hassio/ingress", |     redirect: "/hassio/ingress", | ||||||
| @@ -124,6 +127,14 @@ class HassioMyRedirect extends LitElement { | |||||||
|       } |       } | ||||||
|       resultParams[key] = params[key]; |       resultParams[key] = params[key]; | ||||||
|     }); |     }); | ||||||
|  |     Object.entries(redirect.optional_params || {}).forEach(([key, type]) => { | ||||||
|  |       if (params[key]) { | ||||||
|  |         if (!this._checkParamType(type, params[key])) { | ||||||
|  |           throw Error(); | ||||||
|  |         } | ||||||
|  |         resultParams[key] = params[key]; | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|     return `?${createSearchParam(resultParams)}`; |     return `?${createSearchParam(resultParams)}`; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,7 +8,10 @@ import { atLeastVersion } from "../../src/common/config/version"; | |||||||
| import type { PageNavigation } from "../../src/layouts/hass-tabs-subpage"; | import type { PageNavigation } from "../../src/layouts/hass-tabs-subpage"; | ||||||
| import { HomeAssistant } from "../../src/types"; | import { HomeAssistant } from "../../src/types"; | ||||||
|  |  | ||||||
| export const supervisorTabs = (hass: HomeAssistant): PageNavigation[] => [ | export const supervisorTabs = (hass: HomeAssistant): PageNavigation[] => | ||||||
|  |   atLeastVersion(hass.config.version, 2022, 5) | ||||||
|  |     ? [] | ||||||
|  |     : [ | ||||||
|         { |         { | ||||||
|           translationKey: atLeastVersion(hass.config.version, 2021, 12) |           translationKey: atLeastVersion(hass.config.version, 2021, 12) | ||||||
|             ? "panel.addons" |             ? "panel.addons" | ||||||
| @@ -28,4 +31,4 @@ export const supervisorTabs = (hass: HomeAssistant): PageNavigation[] => [ | |||||||
|           path: `/hassio/system`, |           path: `/hassio/system`, | ||||||
|           iconPath: mdiCogs, |           iconPath: mdiCogs, | ||||||
|         }, |         }, | ||||||
| ]; |       ]; | ||||||
|   | |||||||
| @@ -48,7 +48,7 @@ class HassioCoreInfo extends LitElement { | |||||||
|     ]; |     ]; | ||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-card header="Core"> |       <ha-card header="Core" outlined> | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|           <div> |           <div> | ||||||
|             <ha-settings-row> |             <ha-settings-row> | ||||||
|   | |||||||
| @@ -66,7 +66,7 @@ class HassioHostInfo extends LitElement { | |||||||
|       }, |       }, | ||||||
|     ]; |     ]; | ||||||
|     return html` |     return html` | ||||||
|       <ha-card header="Host"> |       <ha-card header="Host" outlined> | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|           <div> |           <div> | ||||||
|             ${this.supervisor.host.features.includes("hostname") |             ${this.supervisor.host.features.includes("hostname") | ||||||
|   | |||||||
| @@ -23,6 +23,10 @@ import { | |||||||
|   showAlertDialog, |   showAlertDialog, | ||||||
|   showConfirmationDialog, |   showConfirmationDialog, | ||||||
| } from "../../../src/dialogs/generic/show-dialog-box"; | } from "../../../src/dialogs/generic/show-dialog-box"; | ||||||
|  | import { | ||||||
|  |   UNHEALTHY_REASON_URL, | ||||||
|  |   UNSUPPORTED_REASON_URL, | ||||||
|  | } from "../../../src/panels/config/system-health/ha-config-system-health"; | ||||||
| import { haStyle } from "../../../src/resources/styles"; | import { haStyle } from "../../../src/resources/styles"; | ||||||
| import { HomeAssistant } from "../../../src/types"; | import { HomeAssistant } from "../../../src/types"; | ||||||
| import { bytesToString } from "../../../src/util/bytes-to-string"; | import { bytesToString } from "../../../src/util/bytes-to-string"; | ||||||
| @@ -30,11 +34,6 @@ import { documentationUrl } from "../../../src/util/documentation-url"; | |||||||
| import "../components/supervisor-metric"; | import "../components/supervisor-metric"; | ||||||
| import { hassioStyle } from "../resources/hassio-style"; | import { hassioStyle } from "../resources/hassio-style"; | ||||||
|  |  | ||||||
| const UNSUPPORTED_REASON_URL = {}; |  | ||||||
| const UNHEALTHY_REASON_URL = { |  | ||||||
|   privileged: "/more-info/unsupported/privileged", |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| @customElement("hassio-supervisor-info") | @customElement("hassio-supervisor-info") | ||||||
| class HassioSupervisorInfo extends LitElement { | class HassioSupervisorInfo extends LitElement { | ||||||
|   @property({ attribute: false }) public hass!: HomeAssistant; |   @property({ attribute: false }) public hass!: HomeAssistant; | ||||||
| @@ -58,7 +57,7 @@ class HassioSupervisorInfo extends LitElement { | |||||||
|       }, |       }, | ||||||
|     ]; |     ]; | ||||||
|     return html` |     return html` | ||||||
|       <ha-card header="Supervisor"> |       <ha-card header="Supervisor" outlined> | ||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|           <div> |           <div> | ||||||
|             <ha-settings-row> |             <ha-settings-row> | ||||||
|   | |||||||
| @@ -1,16 +1,17 @@ | |||||||
|  | import "../../../src/components/ha-ansi-to-html"; | ||||||
| import "@material/mwc-button"; | import "@material/mwc-button"; | ||||||
| import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import "../../../src/components/buttons/ha-progress-button"; | import "../../../src/components/buttons/ha-progress-button"; | ||||||
| import "../../../src/components/ha-alert"; | import "../../../src/components/ha-alert"; | ||||||
| import "../../../src/components/ha-card"; | import "../../../src/components/ha-card"; | ||||||
|  | import "../../../src/components/ha-select"; | ||||||
| import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | import { extractApiErrorMessage } from "../../../src/data/hassio/common"; | ||||||
| import { fetchHassioLogs } from "../../../src/data/hassio/supervisor"; | import { fetchHassioLogs } from "../../../src/data/hassio/supervisor"; | ||||||
| import { Supervisor } from "../../../src/data/supervisor/supervisor"; | import { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||||
| import "../../../src/layouts/hass-loading-screen"; | import "../../../src/layouts/hass-loading-screen"; | ||||||
| import { haStyle } from "../../../src/resources/styles"; | import { haStyle } from "../../../src/resources/styles"; | ||||||
| import { HomeAssistant } from "../../../src/types"; | import { HomeAssistant } from "../../../src/types"; | ||||||
| import "../components/hassio-ansi-to-html"; |  | ||||||
| import { hassioStyle } from "../resources/hassio-style"; | import { hassioStyle } from "../resources/hassio-style"; | ||||||
|  |  | ||||||
| interface LogProvider { | interface LogProvider { | ||||||
| @@ -64,13 +65,13 @@ class HassioSupervisorLog extends LitElement { | |||||||
|  |  | ||||||
|   protected render(): TemplateResult | void { |   protected render(): TemplateResult | void { | ||||||
|     return html` |     return html` | ||||||
|       <ha-card> |       <ha-card outlined> | ||||||
|         ${this._error |         ${this._error | ||||||
|           ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` |           ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||||
|           : ""} |           : ""} | ||||||
|         ${this.hass.userData?.showAdvanced |         ${this.hass.userData?.showAdvanced | ||||||
|           ? html` |           ? html` | ||||||
|               <mwc-select |               <ha-select | ||||||
|                 .label=${this.supervisor.localize("system.log.log_provider")} |                 .label=${this.supervisor.localize("system.log.log_provider")} | ||||||
|                 @selected=${this._setLogProvider} |                 @selected=${this._setLogProvider} | ||||||
|                 .value=${this._selectedLogProvider} |                 .value=${this._selectedLogProvider} | ||||||
| @@ -82,14 +83,14 @@ class HassioSupervisorLog extends LitElement { | |||||||
|                     </mwc-list-item> |                     </mwc-list-item> | ||||||
|                   ` |                   ` | ||||||
|                 )} |                 )} | ||||||
|               </mwc-select> |               </ha-select> | ||||||
|             ` |             ` | ||||||
|           : ""} |           : ""} | ||||||
|  |  | ||||||
|         <div class="card-content" id="content"> |         <div class="card-content" id="content"> | ||||||
|           ${this._content |           ${this._content | ||||||
|             ? html`<hassio-ansi-to-html .content=${this._content}> |             ? html`<ha-ansi-to-html .content=${this._content}> | ||||||
|               </hassio-ansi-to-html>` |               </ha-ansi-to-html>` | ||||||
|             : html`<hass-loading-screen no-toolbar></hass-loading-screen>`} |             : html`<hass-loading-screen no-toolbar></hass-loading-screen>`} | ||||||
|         </div> |         </div> | ||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
| @@ -145,7 +146,7 @@ class HassioSupervisorLog extends LitElement { | |||||||
|         pre { |         pre { | ||||||
|           white-space: pre-wrap; |           white-space: pre-wrap; | ||||||
|         } |         } | ||||||
|         mwc-select { |         ha-select { | ||||||
|           width: 100%; |           width: 100%; | ||||||
|           margin-bottom: 4px; |           margin-bottom: 4px; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -45,7 +45,6 @@ import { showAlertDialog } from "../../../src/dialogs/generic/show-dialog-box"; | |||||||
| import "../../../src/layouts/hass-loading-screen"; | import "../../../src/layouts/hass-loading-screen"; | ||||||
| import "../../../src/layouts/hass-subpage"; | import "../../../src/layouts/hass-subpage"; | ||||||
| import "../../../src/layouts/hass-tabs-subpage"; | import "../../../src/layouts/hass-tabs-subpage"; | ||||||
| import { SUPERVISOR_UPDATE_NAMES } from "../../../src/panels/config/dashboard/ha-config-updates"; |  | ||||||
| import { HomeAssistant, Route } from "../../../src/types"; | import { HomeAssistant, Route } from "../../../src/types"; | ||||||
| import { addonArchIsSupported, extractChangelog } from "../util/addon"; | import { addonArchIsSupported, extractChangelog } from "../util/addon"; | ||||||
|  |  | ||||||
| @@ -55,6 +54,12 @@ declare global { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | const SUPERVISOR_UPDATE_NAMES = { | ||||||
|  |   core: "Home Assistant Core", | ||||||
|  |   os: "Home Assistant Operating System", | ||||||
|  |   supervisor: "Home Assistant Supervisor", | ||||||
|  | }; | ||||||
|  |  | ||||||
| type updateType = "os" | "supervisor" | "core" | "addon"; | type updateType = "os" | "supervisor" | "core" | "addon"; | ||||||
|  |  | ||||||
| const changelogUrl = ( | const changelogUrl = ( | ||||||
| @@ -123,6 +128,7 @@ class UpdateAvailableCard extends LitElement { | |||||||
|  |  | ||||||
|     return html` |     return html` | ||||||
|       <ha-card |       <ha-card | ||||||
|  |         outlined | ||||||
|         .header=${this.supervisor.localize("update_available.update_name", { |         .header=${this.supervisor.localize("update_available.update_name", { | ||||||
|           name: this._name, |           name: this._name, | ||||||
|         })} |         })} | ||||||
|   | |||||||
							
								
								
									
										36
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,8 +1,8 @@ | |||||||
| { | { | ||||||
|   "description": "A frontend for Home Assistant using the Polymer framework", |   "description": "A frontend for Home Assistant", | ||||||
|   "repository": { |   "repository": { | ||||||
|     "type": "git", |     "type": "git", | ||||||
|     "url": "https://github.com/home-assistant/home-assistant-polymer" |     "url": "https://github.com/home-assistant/frontend" | ||||||
|   }, |   }, | ||||||
|   "name": "home-assistant-frontend", |   "name": "home-assistant-frontend", | ||||||
|   "version": "1.0.0", |   "version": "1.0.0", | ||||||
| @@ -46,6 +46,7 @@ | |||||||
|     "@fullcalendar/daygrid": "5.9.0", |     "@fullcalendar/daygrid": "5.9.0", | ||||||
|     "@fullcalendar/interaction": "5.9.0", |     "@fullcalendar/interaction": "5.9.0", | ||||||
|     "@fullcalendar/list": "5.9.0", |     "@fullcalendar/list": "5.9.0", | ||||||
|  |     "@lit-labs/motion": "^1.0.2", | ||||||
|     "@lit-labs/virtualizer": "patch:@lit-labs/virtualizer@0.7.0-pre.2#./.yarn/patches/@lit-labs/virtualizer/event-target-shim.patch", |     "@lit-labs/virtualizer": "patch:@lit-labs/virtualizer@0.7.0-pre.2#./.yarn/patches/@lit-labs/virtualizer/event-target-shim.patch", | ||||||
|     "@material/chips": "14.0.0-canary.261f2db59.0", |     "@material/chips": "14.0.0-canary.261f2db59.0", | ||||||
|     "@material/data-table": "14.0.0-canary.261f2db59.0", |     "@material/data-table": "14.0.0-canary.261f2db59.0", | ||||||
| @@ -71,14 +72,13 @@ | |||||||
|     "@material/mwc-textfield": "0.25.3", |     "@material/mwc-textfield": "0.25.3", | ||||||
|     "@material/mwc-top-app-bar-fixed": "^0.25.3", |     "@material/mwc-top-app-bar-fixed": "^0.25.3", | ||||||
|     "@material/top-app-bar": "14.0.0-canary.261f2db59.0", |     "@material/top-app-bar": "14.0.0-canary.261f2db59.0", | ||||||
|     "@mdi/js": "6.5.95", |     "@mdi/js": "6.7.96", | ||||||
|     "@mdi/svg": "6.5.95", |     "@mdi/svg": "6.7.96", | ||||||
|     "@polymer/app-layout": "^3.1.0", |     "@polymer/app-layout": "^3.1.0", | ||||||
|     "@polymer/iron-flex-layout": "^3.0.1", |     "@polymer/iron-flex-layout": "^3.0.1", | ||||||
|     "@polymer/iron-icon": "^3.0.1", |     "@polymer/iron-icon": "^3.0.1", | ||||||
|     "@polymer/iron-input": "^3.0.1", |     "@polymer/iron-input": "^3.0.1", | ||||||
|     "@polymer/iron-resizable-behavior": "^3.0.1", |     "@polymer/iron-resizable-behavior": "^3.0.1", | ||||||
|     "@polymer/paper-dropdown-menu": "^3.2.0", |  | ||||||
|     "@polymer/paper-input": "^3.2.1", |     "@polymer/paper-input": "^3.2.1", | ||||||
|     "@polymer/paper-item": "^3.0.1", |     "@polymer/paper-item": "^3.0.1", | ||||||
|     "@polymer/paper-listbox": "^3.0.1", |     "@polymer/paper-listbox": "^3.0.1", | ||||||
| @@ -108,7 +108,7 @@ | |||||||
|     "fuse.js": "^6.0.0", |     "fuse.js": "^6.0.0", | ||||||
|     "google-timezones-json": "^1.0.2", |     "google-timezones-json": "^1.0.2", | ||||||
|     "hls.js": "^1.1.5", |     "hls.js": "^1.1.5", | ||||||
|     "home-assistant-js-websocket": "^6.0.1", |     "home-assistant-js-websocket": "^7.0.3", | ||||||
|     "idb-keyval": "^5.1.3", |     "idb-keyval": "^5.1.3", | ||||||
|     "intl-messageformat": "^9.9.1", |     "intl-messageformat": "^9.9.1", | ||||||
|     "js-yaml": "^4.1.0", |     "js-yaml": "^4.1.0", | ||||||
| @@ -116,7 +116,7 @@ | |||||||
|     "leaflet-draw": "^1.0.4", |     "leaflet-draw": "^1.0.4", | ||||||
|     "lit": "^2.1.2", |     "lit": "^2.1.2", | ||||||
|     "lit-vaadin-helpers": "^0.3.0", |     "lit-vaadin-helpers": "^0.3.0", | ||||||
|     "marked": "^3.0.2", |     "marked": "^4.0.12", | ||||||
|     "memoize-one": "^5.2.1", |     "memoize-one": "^5.2.1", | ||||||
|     "node-vibrant": "3.2.1-alpha.1", |     "node-vibrant": "3.2.1-alpha.1", | ||||||
|     "proxy-polyfill": "^0.3.2", |     "proxy-polyfill": "^0.3.2", | ||||||
| @@ -135,13 +135,12 @@ | |||||||
|     "vis-network": "^8.5.4", |     "vis-network": "^8.5.4", | ||||||
|     "vue": "^2.6.12", |     "vue": "^2.6.12", | ||||||
|     "vue2-daterange-picker": "^0.5.1", |     "vue2-daterange-picker": "^0.5.1", | ||||||
|     "web-animations-js": "^2.3.2", |     "workbox-cacheable-response": "^6.4.2", | ||||||
|     "workbox-cacheable-response": "^6.1.5", |     "workbox-core": "^6.4.2", | ||||||
|     "workbox-core": "^6.1.5", |     "workbox-expiration": "^6.4.2", | ||||||
|     "workbox-expiration": "^6.1.5", |     "workbox-precaching": "^6.4.2", | ||||||
|     "workbox-precaching": "^6.1.5", |     "workbox-routing": "^6.4.2", | ||||||
|     "workbox-routing": "^6.1.5", |     "workbox-strategies": "^6.4.2", | ||||||
|     "workbox-strategies": "^6.1.5", |  | ||||||
|     "xss": "^1.0.9" |     "xss": "^1.0.9" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
| @@ -170,7 +169,7 @@ | |||||||
|     "@types/js-yaml": "^4", |     "@types/js-yaml": "^4", | ||||||
|     "@types/leaflet": "^1", |     "@types/leaflet": "^1", | ||||||
|     "@types/leaflet-draw": "^1", |     "@types/leaflet-draw": "^1", | ||||||
|     "@types/marked": "^2", |     "@types/marked": "^4", | ||||||
|     "@types/mocha": "^8", |     "@types/mocha": "^8", | ||||||
|     "@types/qrcode": "^1.4.2", |     "@types/qrcode": "^1.4.2", | ||||||
|     "@types/sortablejs": "^1", |     "@types/sortablejs": "^1", | ||||||
| @@ -197,7 +196,7 @@ | |||||||
|     "fs-extra": "^7.0.1", |     "fs-extra": "^7.0.1", | ||||||
|     "glob": "^7.2.0", |     "glob": "^7.2.0", | ||||||
|     "gulp": "^4.0.2", |     "gulp": "^4.0.2", | ||||||
|     "gulp-foreach": "^0.1.0", |     "gulp-flatmap": "^1.0.2", | ||||||
|     "gulp-json-transform": "^0.4.6", |     "gulp-json-transform": "^0.4.6", | ||||||
|     "gulp-merge-json": "^1.3.1", |     "gulp-merge-json": "^1.3.1", | ||||||
|     "gulp-rename": "^2.0.0", |     "gulp-rename": "^2.0.0", | ||||||
| @@ -234,7 +233,7 @@ | |||||||
|     "webpack-dev-server": "^4.3.0", |     "webpack-dev-server": "^4.3.0", | ||||||
|     "webpack-manifest-plugin": "^4.0.2", |     "webpack-manifest-plugin": "^4.0.2", | ||||||
|     "webpackbar": "^5.0.0-3", |     "webpackbar": "^5.0.0-3", | ||||||
|     "workbox-build": "^6.1.5" |     "workbox-build": "^6.4.2" | ||||||
|   }, |   }, | ||||||
|   "_comment": "Polymer 3.2 contained a bug, fixed in https://github.com/Polymer/polymer/pull/5569, add as patch", |   "_comment": "Polymer 3.2 contained a bug, fixed in https://github.com/Polymer/polymer/pull/5569, add as patch", | ||||||
|   "resolutions": { |   "resolutions": { | ||||||
| @@ -254,5 +253,6 @@ | |||||||
|   "prettier": { |   "prettier": { | ||||||
|     "trailingComma": "es5", |     "trailingComma": "es5", | ||||||
|     "arrowParens": "always" |     "arrowParens": "always" | ||||||
|   } |   }, | ||||||
|  |   "packageManager": "yarn@3.2.0" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,6 +2,6 @@ | |||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  |  | ||||||
|  |  | ||||||
| def where(): | def where() -> Path: | ||||||
|     """Return path to the frontend.""" |     """Return path to the frontend.""" | ||||||
|     return Path(__file__).parent |     return Path(__file__).parent | ||||||
|   | |||||||
							
								
								
									
										0
									
								
								public/py.typed
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								public/py.typed
									
									
									
									
									
										Normal file
									
								
							| @@ -1,3 +1,30 @@ | |||||||
| [build-system] | [build-system] | ||||||
| requires = ["setuptools~=60.5", "wheel~=0.37.1"] | requires = ["setuptools~=62.3", "wheel~=0.37.1"] | ||||||
| build-backend = "setuptools.build_meta" | build-backend = "setuptools.build_meta" | ||||||
|  |  | ||||||
|  | [project] | ||||||
|  | name         = "home-assistant-frontend" | ||||||
|  | version      = "20220525.0" | ||||||
|  | license      = {text = "Apache-2.0"} | ||||||
|  | description  = "The Home Assistant frontend" | ||||||
|  | readme       = "README.md" | ||||||
|  | authors      = [ | ||||||
|  |     {name = "The Home Assistant Authors", email = "hello@home-assistant.io"} | ||||||
|  | ] | ||||||
|  | requires-python = ">=3.4.0" | ||||||
|  |  | ||||||
|  | [project.urls] | ||||||
|  | "Homepage" = "https://github.com/home-assistant/frontend" | ||||||
|  |  | ||||||
|  | [tool.setuptools] | ||||||
|  | platforms = ["any"] | ||||||
|  | zip-safe  = false | ||||||
|  | include-package-data = true | ||||||
|  |  | ||||||
|  | [tool.setuptools.packages.find] | ||||||
|  | include = ["hass_frontend*"] | ||||||
|  |  | ||||||
|  | [tool.mypy] | ||||||
|  | python_version = 3.4 | ||||||
|  | show_error_codes = true | ||||||
|  | strict = true | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ if [ -z $(which hass) ]; then | |||||||
|   echo "Installing Home Asstant core from dev." |   echo "Installing Home Asstant core from dev." | ||||||
|   python3 -m pip install --upgrade \ |   python3 -m pip install --upgrade \ | ||||||
|     colorlog \ |     colorlog \ | ||||||
|     git+git://github.com/home-assistant/home-assistant.git@dev |     git+https://github.com/home-assistant/home-assistant.git@dev | ||||||
| fi | fi | ||||||
|  |  | ||||||
| if [ ! -d "${WD}/config" ]; then | if [ ! -d "${WD}/config" ]; then | ||||||
|   | |||||||
| @@ -50,14 +50,14 @@ async function main(args) { | |||||||
|     return; |     return; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   const setup = fs.readFileSync("setup.cfg", "utf8"); |   const setup = fs.readFileSync("pyproject.toml", "utf8"); | ||||||
|   const version = setup.match(/\d{8}\.\d+/)[0]; |   const version = setup.match(/version\W+=\W"(\d{8}\.\d)"/)[1]; | ||||||
|   const newVersion = method(version); |   const newVersion = method(version); | ||||||
|  |  | ||||||
|   console.log("Current version:", version); |   console.log("Current version:", version); | ||||||
|   console.log("New version:", newVersion); |   console.log("New version:", newVersion); | ||||||
|  |  | ||||||
|   fs.writeFileSync("setup.cfg", setup.replace(version, newVersion), "utf-8"); |   fs.writeFileSync("pyproject.toml", setup.replace(version, newVersion), "utf-8"); | ||||||
|  |  | ||||||
|   if (!commit) { |   if (!commit) { | ||||||
|     return; |     return; | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								setup.cfg
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								setup.cfg
									
									
									
									
									
								
							| @@ -1,21 +0,0 @@ | |||||||
| [metadata] |  | ||||||
| name         = home-assistant-frontend |  | ||||||
| version      = 20220222.0 |  | ||||||
| author       = The Home Assistant Authors |  | ||||||
| author_email = hello@home-assistant.io |  | ||||||
| license      = Apache-2.0 |  | ||||||
| platforms    = any |  | ||||||
| description  = The Home Assistant frontend |  | ||||||
| long_description = file: README.md |  | ||||||
| long_description_content_type = text/markdown |  | ||||||
| url          = https://github.com/home-assistant/frontend |  | ||||||
|  |  | ||||||
| [options] |  | ||||||
| packages = find: |  | ||||||
| zip_safe = False |  | ||||||
| include_package_data = True |  | ||||||
| python_requires = >= 3.4.0 |  | ||||||
|  |  | ||||||
| [options.packages.find] |  | ||||||
| include = |  | ||||||
|     hass_frontend* |  | ||||||
							
								
								
									
										7
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										7
									
								
								setup.py
									
									
									
									
									
								
							| @@ -1,7 +0,0 @@ | |||||||
| """ |  | ||||||
| Entry point for setuptools. Required for editable installs. |  | ||||||
| TODO: Remove file after updating to pip 21.3 |  | ||||||
| """ |  | ||||||
| from setuptools import setup |  | ||||||
|  |  | ||||||
| setup() |  | ||||||
| @@ -101,13 +101,19 @@ class HaAuthorize extends litLocalizeLiteMixin(LitElement) { | |||||||
|     this._fetchAuthProviders(); |     this._fetchAuthProviders(); | ||||||
|  |  | ||||||
|     if (matchMedia("(prefers-color-scheme: dark)").matches) { |     if (matchMedia("(prefers-color-scheme: dark)").matches) { | ||||||
|       applyThemesOnElement(document.documentElement, { |       applyThemesOnElement( | ||||||
|  |         document.documentElement, | ||||||
|  |         { | ||||||
|           default_theme: "default", |           default_theme: "default", | ||||||
|           default_dark_theme: null, |           default_dark_theme: null, | ||||||
|           themes: {}, |           themes: {}, | ||||||
|           darkMode: true, |           darkMode: true, | ||||||
|           theme: "default", |           theme: "default", | ||||||
|       }); |         }, | ||||||
|  |         undefined, | ||||||
|  |         undefined, | ||||||
|  |         true | ||||||
|  |       ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (!this.redirectUri) { |     if (!this.redirectUri) { | ||||||
|   | |||||||
| @@ -187,6 +187,7 @@ export const DOMAINS_WITH_MORE_INFO = [ | |||||||
|   "scene", |   "scene", | ||||||
|   "sun", |   "sun", | ||||||
|   "timer", |   "timer", | ||||||
|  |   "update", | ||||||
|   "vacuum", |   "vacuum", | ||||||
|   "water_heater", |   "water_heater", | ||||||
|   "weather", |   "weather", | ||||||
| @@ -200,6 +201,7 @@ export const DOMAINS_HIDE_DEFAULT_MORE_INFO = [ | |||||||
|   "input_text", |   "input_text", | ||||||
|   "number", |   "number", | ||||||
|   "scene", |   "scene", | ||||||
|  |   "update", | ||||||
|   "select", |   "select", | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,9 +3,9 @@ import type { ForDict } from "../../data/automation"; | |||||||
|  |  | ||||||
| export const createDurationData = ( | export const createDurationData = ( | ||||||
|   duration: string | number | ForDict | undefined |   duration: string | number | ForDict | undefined | ||||||
| ): HaDurationData => { | ): HaDurationData | undefined => { | ||||||
|   if (duration === undefined) { |   if (duration === undefined) { | ||||||
|     return {}; |     return undefined; | ||||||
|   } |   } | ||||||
|   if (typeof duration !== "object") { |   if (typeof duration !== "object") { | ||||||
|     if (typeof duration === "string" || isNaN(duration)) { |     if (typeof duration === "string" || isNaN(duration)) { | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								src/common/datetime/duration.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/common/datetime/duration.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | import secondsToDuration from "./seconds_to_duration"; | ||||||
|  |  | ||||||
|  | const DAY_IN_SECONDS = 86400; | ||||||
|  | const HOUR_IN_SECONDS = 3600; | ||||||
|  | const MINUTE_IN_SECONDS = 60; | ||||||
|  |  | ||||||
|  | export const UNIT_TO_SECOND_CONVERT = { | ||||||
|  |   s: 1, | ||||||
|  |   min: MINUTE_IN_SECONDS, | ||||||
|  |   h: HOUR_IN_SECONDS, | ||||||
|  |   d: DAY_IN_SECONDS, | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const formatDuration = (duration: string, units: string): string => | ||||||
|  |   secondsToDuration(parseFloat(duration) * UNIT_TO_SECOND_CONVERT[units]) || | ||||||
|  |   "0"; | ||||||
							
								
								
									
										41
									
								
								src/common/dom/ancestors-with-property.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								src/common/dom/ancestors-with-property.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | |||||||
|  | const DEFAULT_OWN = true; | ||||||
|  |  | ||||||
|  | // Finds the closest ancestor of an element that has a specific optionally owned property, | ||||||
|  | // traversing slot and shadow root boundaries until the body element is reached | ||||||
|  | export const closestWithProperty = ( | ||||||
|  |   element: Element | null, | ||||||
|  |   property: string | symbol, | ||||||
|  |   own = DEFAULT_OWN | ||||||
|  | ) => { | ||||||
|  |   if (!element || element === document.body) return null; | ||||||
|  |  | ||||||
|  |   element = element.assignedSlot ?? element; | ||||||
|  |   if (element.parentElement) { | ||||||
|  |     element = element.parentElement; | ||||||
|  |   } else { | ||||||
|  |     const root = element.getRootNode(); | ||||||
|  |     element = root instanceof ShadowRoot ? root.host : null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   if ( | ||||||
|  |     own | ||||||
|  |       ? Object.prototype.hasOwnProperty.call(element, property) | ||||||
|  |       : element && property in element | ||||||
|  |   ) | ||||||
|  |     return element; | ||||||
|  |   return closestWithProperty(element, property, own); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Finds the set of all such ancestors and includes starting element as first in the set | ||||||
|  | export const ancestorsWithProperty = ( | ||||||
|  |   element: Element | null, | ||||||
|  |   property: string | symbol, | ||||||
|  |   own = DEFAULT_OWN | ||||||
|  | ) => { | ||||||
|  |   const ancestors: Set<Element> = new Set(); | ||||||
|  |   while (element) { | ||||||
|  |     ancestors.add(element); | ||||||
|  |     element = closestWithProperty(element, property, own); | ||||||
|  |   } | ||||||
|  |   return ancestors; | ||||||
|  | }; | ||||||
| @@ -31,11 +31,12 @@ export const applyThemesOnElement = ( | |||||||
|   element, |   element, | ||||||
|   themes: HomeAssistant["themes"], |   themes: HomeAssistant["themes"], | ||||||
|   selectedTheme?: string, |   selectedTheme?: string, | ||||||
|   themeSettings?: Partial<HomeAssistant["selectedTheme"]> |   themeSettings?: Partial<HomeAssistant["selectedTheme"]>, | ||||||
|  |   main?: boolean | ||||||
| ) => { | ) => { | ||||||
|   // If there is no explicitly desired theme provided, we automatically |   // If there is no explicitly desired theme provided, and the element is the main element we automatically | ||||||
|   // use the active one from `themes`. |   // use the active one from `themes`. | ||||||
|   const themeToApply = selectedTheme || themes.theme; |   const themeToApply = selectedTheme || (main ? themes.theme : undefined); | ||||||
|  |  | ||||||
|   // If there is no explicitly desired dark mode provided, we automatically |   // If there is no explicitly desired dark mode provided, we automatically | ||||||
|   // use the active one from `themes`. |   // use the active one from `themes`. | ||||||
| @@ -47,7 +48,7 @@ export const applyThemesOnElement = ( | |||||||
|   let cacheKey = themeToApply; |   let cacheKey = themeToApply; | ||||||
|   let themeRules: Partial<ThemeVars> = {}; |   let themeRules: Partial<ThemeVars> = {}; | ||||||
|  |  | ||||||
|   if (darkMode) { |   if (themeToApply && darkMode) { | ||||||
|     cacheKey = `${cacheKey}__dark`; |     cacheKey = `${cacheKey}__dark`; | ||||||
|     themeRules = { ...darkStyles }; |     themeRules = { ...darkStyles }; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -12,7 +12,7 @@ export const isNavigationClick = (e: MouseEvent) => { | |||||||
|  |  | ||||||
|   const anchor = e |   const anchor = e | ||||||
|     .composedPath() |     .composedPath() | ||||||
|     .filter((n) => (n as HTMLElement).tagName === "A")[0] as |     .find((n) => (n as HTMLElement).tagName === "A") as | ||||||
|     | HTMLAnchorElement |     | HTMLAnchorElement | ||||||
|     | undefined; |     | undefined; | ||||||
|   if ( |   if ( | ||||||
|   | |||||||
| @@ -29,8 +29,11 @@ import { | |||||||
|   mdiPowerPlug, |   mdiPowerPlug, | ||||||
|   mdiPowerPlugOff, |   mdiPowerPlugOff, | ||||||
|   mdiRadioboxBlank, |   mdiRadioboxBlank, | ||||||
|   mdiSmoke, |  | ||||||
|   mdiSnowflake, |   mdiSnowflake, | ||||||
|  |   mdiSmokeDetector, | ||||||
|  |   mdiSmokeDetectorAlert, | ||||||
|  |   mdiSmokeDetectorVariant, | ||||||
|  |   mdiSmokeDetectorVariantAlert, | ||||||
|   mdiSquare, |   mdiSquare, | ||||||
|   mdiSquareOutline, |   mdiSquareOutline, | ||||||
|   mdiStop, |   mdiStop, | ||||||
| @@ -52,6 +55,8 @@ export const binarySensorIcon = (state?: string, stateObj?: HassEntity) => { | |||||||
|       return is_off ? mdiBattery : mdiBatteryOutline; |       return is_off ? mdiBattery : mdiBatteryOutline; | ||||||
|     case "battery_charging": |     case "battery_charging": | ||||||
|       return is_off ? mdiBattery : mdiBatteryCharging; |       return is_off ? mdiBattery : mdiBatteryCharging; | ||||||
|  |     case "carbon_monoxide": | ||||||
|  |       return is_off ? mdiSmokeDetector : mdiSmokeDetectorAlert; | ||||||
|     case "cold": |     case "cold": | ||||||
|       return is_off ? mdiThermometer : mdiSnowflake; |       return is_off ? mdiThermometer : mdiSnowflake; | ||||||
|     case "connectivity": |     case "connectivity": | ||||||
| @@ -68,7 +73,7 @@ export const binarySensorIcon = (state?: string, stateObj?: HassEntity) => { | |||||||
|     case "tamper": |     case "tamper": | ||||||
|       return is_off ? mdiCheckCircle : mdiAlertCircle; |       return is_off ? mdiCheckCircle : mdiAlertCircle; | ||||||
|     case "smoke": |     case "smoke": | ||||||
|       return is_off ? mdiCheckCircle : mdiSmoke; |       return is_off ? mdiSmokeDetectorVariant : mdiSmokeDetectorVariantAlert; | ||||||
|     case "heat": |     case "heat": | ||||||
|       return is_off ? mdiThermometer : mdiFire; |       return is_off ? mdiThermometer : mdiFire; | ||||||
|     case "light": |     case "light": | ||||||
|   | |||||||
| @@ -1,6 +1,11 @@ | |||||||
| import { HassEntity } from "home-assistant-js-websocket"; | import { HassEntity } from "home-assistant-js-websocket"; | ||||||
|  | import { UNAVAILABLE_STATES } from "../../data/entity"; | ||||||
|  |  | ||||||
| export const computeActiveState = (stateObj: HassEntity): string => { | export const computeActiveState = (stateObj: HassEntity): string => { | ||||||
|  |   if (UNAVAILABLE_STATES.includes(stateObj.state)) { | ||||||
|  |     return stateObj.state; | ||||||
|  |   } | ||||||
|  |  | ||||||
|   const domain = stateObj.entity_id.split(".")[0]; |   const domain = stateObj.entity_id.split(".")[0]; | ||||||
|   let state = stateObj.state; |   let state = stateObj.state; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,45 +1,75 @@ | |||||||
| import { HassEntity } from "home-assistant-js-websocket"; | import { HassEntity } from "home-assistant-js-websocket"; | ||||||
| import { UNAVAILABLE, UNKNOWN } from "../../data/entity"; | import { UNAVAILABLE, UNKNOWN } from "../../data/entity"; | ||||||
| import { FrontendLocaleData } from "../../data/translation"; | import { FrontendLocaleData } from "../../data/translation"; | ||||||
|  | import { | ||||||
|  |   UPDATE_SUPPORT_PROGRESS, | ||||||
|  |   updateIsInstallingFromAttributes, | ||||||
|  | } from "../../data/update"; | ||||||
| import { formatDate } from "../datetime/format_date"; | import { formatDate } from "../datetime/format_date"; | ||||||
| import { formatDateTime } from "../datetime/format_date_time"; | import { formatDateTime } from "../datetime/format_date_time"; | ||||||
| import { formatTime } from "../datetime/format_time"; | import { formatTime } from "../datetime/format_time"; | ||||||
| import { formatNumber, isNumericState } from "../number/format_number"; | import { formatNumber, isNumericFromAttributes } from "../number/format_number"; | ||||||
| import { LocalizeFunc } from "../translations/localize"; | import { LocalizeFunc } from "../translations/localize"; | ||||||
| import { computeStateDomain } from "./compute_state_domain"; | import { supportsFeatureFromAttributes } from "./supports-feature"; | ||||||
|  | import { formatDuration, UNIT_TO_SECOND_CONVERT } from "../datetime/duration"; | ||||||
|  | import { computeDomain } from "./compute_domain"; | ||||||
|  |  | ||||||
| export const computeStateDisplay = ( | export const computeStateDisplay = ( | ||||||
|   localize: LocalizeFunc, |   localize: LocalizeFunc, | ||||||
|   stateObj: HassEntity, |   stateObj: HassEntity, | ||||||
|   locale: FrontendLocaleData, |   locale: FrontendLocaleData, | ||||||
|   state?: string |   state?: string | ||||||
| ): string => { | ): string => | ||||||
|   const compareState = state !== undefined ? state : stateObj.state; |   computeStateDisplayFromEntityAttributes( | ||||||
|  |     localize, | ||||||
|  |     locale, | ||||||
|  |     stateObj.entity_id, | ||||||
|  |     stateObj.attributes, | ||||||
|  |     state !== undefined ? state : stateObj.state | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   if (compareState === UNKNOWN || compareState === UNAVAILABLE) { | export const computeStateDisplayFromEntityAttributes = ( | ||||||
|     return localize(`state.default.${compareState}`); |   localize: LocalizeFunc, | ||||||
|  |   locale: FrontendLocaleData, | ||||||
|  |   entityId: string, | ||||||
|  |   attributes: any, | ||||||
|  |   state: string | ||||||
|  | ): string => { | ||||||
|  |   if (state === UNKNOWN || state === UNAVAILABLE) { | ||||||
|  |     return localize(`state.default.${state}`); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Entities with a `unit_of_measurement` or `state_class` are numeric values and should use `formatNumber` |   // Entities with a `unit_of_measurement` or `state_class` are numeric values and should use `formatNumber` | ||||||
|   if (isNumericState(stateObj)) { |   if (isNumericFromAttributes(attributes)) { | ||||||
|     if (stateObj.attributes.device_class === "monetary") { |     // state is duration | ||||||
|  |     if ( | ||||||
|  |       attributes.device_class === "duration" && | ||||||
|  |       attributes.unit_of_measurement && | ||||||
|  |       UNIT_TO_SECOND_CONVERT[attributes.unit_of_measurement] | ||||||
|  |     ) { | ||||||
|       try { |       try { | ||||||
|         return formatNumber(compareState, locale, { |         return formatDuration(state, attributes.unit_of_measurement); | ||||||
|  |       } catch (_err) { | ||||||
|  |         // fallback to default | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     if (attributes.device_class === "monetary") { | ||||||
|  |       try { | ||||||
|  |         return formatNumber(state, locale, { | ||||||
|           style: "currency", |           style: "currency", | ||||||
|           currency: stateObj.attributes.unit_of_measurement, |           currency: attributes.unit_of_measurement, | ||||||
|  |           minimumFractionDigits: 2, | ||||||
|         }); |         }); | ||||||
|       } catch (_err) { |       } catch (_err) { | ||||||
|         // fallback to default |         // fallback to default | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     return `${formatNumber(compareState, locale)}${ |     return `${formatNumber(state, locale)}${ | ||||||
|       stateObj.attributes.unit_of_measurement |       attributes.unit_of_measurement ? " " + attributes.unit_of_measurement : "" | ||||||
|         ? " " + stateObj.attributes.unit_of_measurement |  | ||||||
|         : "" |  | ||||||
|     }`; |     }`; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   const domain = computeStateDomain(stateObj); |   const domain = computeDomain(entityId); | ||||||
|  |  | ||||||
|   if (domain === "input_datetime") { |   if (domain === "input_datetime") { | ||||||
|     if (state !== undefined) { |     if (state !== undefined) { | ||||||
| @@ -74,36 +104,32 @@ export const computeStateDisplay = ( | |||||||
|     } else { |     } else { | ||||||
|       // If not trying to display an explicit state, create `Date` object from `stateObj`'s attributes then format. |       // If not trying to display an explicit state, create `Date` object from `stateObj`'s attributes then format. | ||||||
|       let date: Date; |       let date: Date; | ||||||
|       if (stateObj.attributes.has_date && stateObj.attributes.has_time) { |       if (attributes.has_date && attributes.has_time) { | ||||||
|         date = new Date( |         date = new Date( | ||||||
|           stateObj.attributes.year, |           attributes.year, | ||||||
|           stateObj.attributes.month - 1, |           attributes.month - 1, | ||||||
|           stateObj.attributes.day, |           attributes.day, | ||||||
|           stateObj.attributes.hour, |           attributes.hour, | ||||||
|           stateObj.attributes.minute |           attributes.minute | ||||||
|         ); |         ); | ||||||
|         return formatDateTime(date, locale); |         return formatDateTime(date, locale); | ||||||
|       } |       } | ||||||
|       if (stateObj.attributes.has_date) { |       if (attributes.has_date) { | ||||||
|         date = new Date( |         date = new Date(attributes.year, attributes.month - 1, attributes.day); | ||||||
|           stateObj.attributes.year, |  | ||||||
|           stateObj.attributes.month - 1, |  | ||||||
|           stateObj.attributes.day |  | ||||||
|         ); |  | ||||||
|         return formatDate(date, locale); |         return formatDate(date, locale); | ||||||
|       } |       } | ||||||
|       if (stateObj.attributes.has_time) { |       if (attributes.has_time) { | ||||||
|         date = new Date(); |         date = new Date(); | ||||||
|         date.setHours(stateObj.attributes.hour, stateObj.attributes.minute); |         date.setHours(attributes.hour, attributes.minute); | ||||||
|         return formatTime(date, locale); |         return formatTime(date, locale); | ||||||
|       } |       } | ||||||
|       return stateObj.state; |       return state; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (domain === "humidifier") { |   if (domain === "humidifier") { | ||||||
|     if (compareState === "on" && stateObj.attributes.humidity) { |     if (state === "on" && attributes.humidity) { | ||||||
|       return `${stateObj.attributes.humidity} %`; |       return `${attributes.humidity} %`; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -113,7 +139,7 @@ export const computeStateDisplay = ( | |||||||
|     domain === "number" || |     domain === "number" || | ||||||
|     domain === "input_number" |     domain === "input_number" | ||||||
|   ) { |   ) { | ||||||
|     return formatNumber(compareState, locale); |     return formatNumber(state, locale); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // state of button is a timestamp |   // state of button is a timestamp | ||||||
| @@ -121,24 +147,44 @@ export const computeStateDisplay = ( | |||||||
|     domain === "button" || |     domain === "button" || | ||||||
|     domain === "input_button" || |     domain === "input_button" || | ||||||
|     domain === "scene" || |     domain === "scene" || | ||||||
|     (domain === "sensor" && stateObj.attributes.device_class === "timestamp") |     (domain === "sensor" && attributes.device_class === "timestamp") | ||||||
|   ) { |   ) { | ||||||
|     try { |     try { | ||||||
|       return formatDateTime(new Date(compareState), locale); |       return formatDateTime(new Date(state), locale); | ||||||
|     } catch (_err) { |     } catch (_err) { | ||||||
|       return compareState; |       return state; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   if (domain === "update") { | ||||||
|  |     // When updating, and entity does not support % show "Installing" | ||||||
|  |     // When updating, and entity does support % show "Installing (xx%)" | ||||||
|  |     // When update available, show the version | ||||||
|  |     // When the latest version is skipped, show the latest version | ||||||
|  |     // When update is not available, show "Up-to-date" | ||||||
|  |     // When update is not available and there is no latest_version show "Unavailable" | ||||||
|  |     return state === "on" | ||||||
|  |       ? updateIsInstallingFromAttributes(attributes) | ||||||
|  |         ? supportsFeatureFromAttributes(attributes, UPDATE_SUPPORT_PROGRESS) | ||||||
|  |           ? localize("ui.card.update.installing_with_progress", { | ||||||
|  |               progress: attributes.in_progress, | ||||||
|  |             }) | ||||||
|  |           : localize("ui.card.update.installing") | ||||||
|  |         : attributes.latest_version | ||||||
|  |       : attributes.skipped_version === attributes.latest_version | ||||||
|  |       ? attributes.latest_version ?? localize("state.default.unavailable") | ||||||
|  |       : localize("ui.card.update.up_to_date"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     // Return device class translation |     // Return device class translation | ||||||
|     (stateObj.attributes.device_class && |     (attributes.device_class && | ||||||
|       localize( |       localize( | ||||||
|         `component.${domain}.state.${stateObj.attributes.device_class}.${compareState}` |         `component.${domain}.state.${attributes.device_class}.${state}` | ||||||
|       )) || |       )) || | ||||||
|     // Return default translation |     // Return default translation | ||||||
|     localize(`component.${domain}.state._.${compareState}`) || |     localize(`component.${domain}.state._.${state}`) || | ||||||
|     // We don't know! Return the raw state. |     // We don't know! Return the raw state. | ||||||
|     compareState |     state | ||||||
|   ); |   ); | ||||||
| }; | }; | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user