mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-25 11:39:41 +00:00 
			
		
		
		
	Compare commits
	
		
			1026 Commits
		
	
	
		
			20250701.0
			...
			dropdown
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 9d4bf30753 | ||
|   | 10b99433ea | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 43a23e6cdd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | aa4dd1cf29 | ||
|   | 0ae55c39cc | ||
|   | 0bfacacc9e | ||
|   | c2f21c19af | ||
|   | 6653333c38 | ||
|   | 8c19e080be | ||
|   | c649b1015a | ||
|   | 1b6c33efd4 | ||
|   | 5cfc34b020 | ||
|   | 1e7647b214 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cef3a7ef99 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 14d0028426 | ||
|   | 28032d9d0d | ||
|   | 6c1995ba1b | ||
|   | b68464c5d5 | ||
|   | 31ccf114a6 | ||
|   | 1b932ae4a2 | ||
|   | 0df6019b95 | ||
|   | f0d4c9cb72 | ||
|   | 94fb03d2e2 | ||
|   | 6dc165ebf8 | ||
|   | f2c5b91def | ||
|   | b312cca050 | ||
|   | ac14733bff | ||
|   | a2d4165511 | ||
|   | b87ffbd4f7 | ||
|   | a8f8d197f8 | ||
|   | 4fcac79047 | ||
|   | 42ddacd41a | ||
|   | ebc9981289 | ||
|   | 23deab253b | ||
|   | ab172abe02 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 10d5d8b15d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c9e472dab7 | ||
|   | 1e13b2b812 | ||
|   | e04a04632a | ||
|   | 04bc5fba63 | ||
|   | e66724ca9e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bcfe5add33 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7cc116dd07 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ee93f31220 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b7cc19f12e | ||
|   | f70edf9311 | ||
|   | 0fa7c2face | ||
|   | 7b3a265a70 | ||
|   | 5d9aae3ad5 | ||
|   | 5de84ac0d8 | ||
|   | 98c4ec91d6 | ||
|   | 972b9cb758 | ||
|   | ac621af811 | ||
|   | 7eb97bb58f | ||
|   | d37af0f488 | ||
|   | 0d3b340228 | ||
|   | 288e03775b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | df36e9d205 | ||
|   | 15a0b35866 | ||
|   | aa7522f681 | ||
|   | c09e97a561 | ||
|   | 733be8e5a3 | ||
|   | d107ac7d4c | ||
|   | efc5bacb97 | ||
|   | 430e52efe3 | ||
|   | 6b4c4a9cf8 | ||
|   | e5b1acc2c3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c89f476d67 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e68afead17 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c4651c0bc0 | ||
|   | 6d95b7af11 | ||
|   | 3e74cf3ada | ||
|   | 859ee98abb | ||
|   | dd3e5e3724 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2e3ab4d64f | ||
|   | 63cbeca820 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1057ff314c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5b946f1048 | ||
|   | fdd66b5cec | ||
|   | 76c9723c71 | ||
|   | b02368b9c6 | ||
|   | 0bcb7897c9 | ||
|   | 786bbb3850 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e8ead84fe5 | ||
|   | 428e7fb332 | ||
|   | ad9e8d5a52 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e3cf04b3d1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 10c3042db1 | ||
|   | 25f6b7de2f | ||
|   | ca1cda4824 | ||
|   | 8c4a67315b | ||
|   | c18de97b32 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 23a3ca3ed7 | ||
|   | 69457b4e85 | ||
|   | 2e096c23e0 | ||
|   | 552691e200 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 91258c86c1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3750a378cd | ||
|   | 12d3304c72 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 246100809d | ||
|   | 6efca93186 | ||
|   | 6280647b9a | ||
|   | 2ff52c6c29 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d038e11170 | ||
|   | 8925b39fe5 | ||
|   | beeef65506 | ||
|   | 994c1b5751 | ||
|   | 6823c647b6 | ||
|   | 866b478dc0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d746dc5752 | ||
|   | 5f53e1e71c | ||
|   | 3da82df093 | ||
|   | 4cedfffb71 | ||
|   | 1e1514e7da | ||
|   | 60e07075bc | ||
|   | c998086474 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 53be0a3fa2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d69c46c80c | ||
|   | 0c2a7bfed0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | afdd232e38 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 179751a135 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 52f6024306 | ||
|   | 7c7a4e61f2 | ||
|   | facce7b016 | ||
|   | e546cb3374 | ||
|   | a0d2e7312b | ||
|   | c0a9dadcbe | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e1edf7fb98 | ||
|   | 6d5c165bd2 | ||
|   | 54177a16e9 | ||
|   | c814b8e888 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 33a0b32cc5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7dae13bf57 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0a3fe6e0fb | ||
|   | e0348e4da7 | ||
|   | d53f3ec898 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e422547d93 | ||
|   | d91a3fbe85 | ||
|   | 01d7130f22 | ||
|   | c57851e4df | ||
|   | 6f1f13acb0 | ||
|   | a8abd00809 | ||
|   | e053978dbe | ||
|   | 6e57f726a3 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b7cabadbe1 | ||
|   | d920217374 | ||
|   | 1630263276 | ||
|   | 5680c742be | ||
|   | 2aeb9cf0ef | ||
|   | c9931b3a3c | ||
|   | fbf7ebdfe4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 52ccb03de5 | ||
|   | 900236ac07 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 28940c930d | ||
|   | e278b463fd | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | db2acd4e39 | ||
|   | 6dcc52cd44 | ||
|   | 981db50826 | ||
|   | 09683863a7 | ||
|   | 8c78f931dc | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 40ce3c1e31 | ||
|   | e430a1b1be | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a2c6116417 | ||
|   | 3239273f3e | ||
|   | e42c5a3254 | ||
|   | df7a6297b0 | ||
|   | e4ca478d01 | ||
|   | 7be2c59295 | ||
|   | 99d9c67492 | ||
|   | 8f781e53e3 | ||
|   | 3c92826e71 | ||
|   | 151a879e0a | ||
|   | f3a8529ed7 | ||
|   | d2cc7856d1 | ||
|   | d5cb815bbd | ||
|   | 7f88d863e9 | ||
|   | 88ac56ac0b | ||
|   | 3d173ad03e | ||
|   | 3889d71768 | ||
|   | 8872adf2ed | ||
|   | 969e655fff | ||
|   | cdc913d878 | ||
|   | 4ac1215def | ||
|   | b2376fba56 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f14d9198ac | ||
|   | f4e583b302 | ||
|   | 2c602aecee | ||
|   | cbf96898fe | ||
|   | 6760f4a2ae | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3481f7e8be | ||
|   | 95a0fe335f | ||
|   | 1e2d144d26 | ||
|   | 6aa89cb532 | ||
|   | 1b0ed7017f | ||
|   | 1cc7e387da | ||
|   | 41bf935f6e | ||
|   | b08ea36a1e | ||
|   | 4f52a46725 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f8a82563b0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a1672ccdfb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bde851e5a4 | ||
|   | a6d3041d59 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f64edfa305 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 067b321d84 | ||
|   | 33efe395c8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | db26b1041f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6e9b4637bb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0e30e5e0f4 | ||
|   | 283da74e2d | ||
|   | 034afd1375 | ||
|   | 912d710ae4 | ||
|   | 86b99d931a | ||
|   | 35cfa9aa0d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6a23dbf204 | ||
|   | cef8fc1d38 | ||
|   | 7c06e33b50 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cb365d4635 | ||
|   | 525102678b | ||
|   | dfc4b0bba2 | ||
|   | 846692bc8a | ||
|   | 3b90b5fcb1 | ||
|   | cac978344f | ||
|   | 6a40631e6d | ||
|   | 48f5b6dfd3 | ||
|   | 04b01d2cd9 | ||
|   | 0e8e054db1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 477a893193 | ||
|   | bd0822f09f | ||
|   | 07c3ffb55d | ||
|   | fbfb4709d2 | ||
|   | 0a5b31e328 | ||
|   | 8cf0d8d2c3 | ||
|   | 61c16ce020 | ||
|   | 6bede4ddca | ||
|   | bd88b91071 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 29b02a3c99 | ||
|   | ac87e2280d | ||
|   | 98c4e34a23 | ||
|   | 3d005c8316 | ||
|   | af31b5add3 | ||
|   | 9d02a1d391 | ||
|   | 98e6f32fe8 | ||
|   | 2726c6a849 | ||
|   | c09ec54c76 | ||
|   | 9f045538a2 | ||
|   | c6c4f91b0e | ||
|   | f71d8f4367 | ||
|   | 68c1a38231 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a9796e4216 | ||
|   | bf6eefb692 | ||
|   | 7ec3b08444 | ||
|   | f3355671d1 | ||
|   | c0e240a3bf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 00fd4753e4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 08ac873e3b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d12b8d1b1b | ||
|   | 977207dde4 | ||
|   | 87a5f1a315 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | acab2d5ead | ||
|   | 046fc00f73 | ||
|   | 05775c411b | ||
|   | d64acca598 | ||
|   | 59571d03a6 | ||
|   | 28c515bbac | ||
|   | 27db5b3b02 | ||
|   | 1922db0474 | ||
|   | c8c74a9744 | ||
|   | 2c676baa99 | ||
|   | 3e41474faa | ||
|   | 5f9c69ac21 | ||
|   | 8b45ccaaba | ||
|   | 455925f637 | ||
|   | 9fba7427f8 | ||
|   | 21aae02652 | ||
|   | 24e3fbf622 | ||
|   | dcbc8b627f | ||
|   | 0d8d18617c | ||
|   | 7eb87c78cc | ||
|   | 0eaf9ead9e | ||
|   | 7082646fe5 | ||
|   | 96d364b3bd | ||
|   | e726eb7370 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e6f587da78 | ||
|   | c595392abe | ||
|   | 5bcffd0dbe | ||
|   | df801833fc | ||
|   | 5ba5c00c70 | ||
|   | dcea227f4a | ||
|   | 1abedcd5f0 | ||
|   | 9e29693293 | ||
|   | 3bfafc794f | ||
|   | 89c43b2b33 | ||
|   | 466115d916 | ||
|   | a34ca3c085 | ||
|   | 9a8ca36047 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b454e89613 | ||
|   | 0b76109272 | ||
|   | 942d264693 | ||
|   | b10fdf8438 | ||
|   | bee8980192 | ||
|   | 61487565db | ||
|   | cc70eb46c9 | ||
|   | dec9d304da | ||
|   | 7f8e856102 | ||
|   | 4bd60a1366 | ||
|   | e9ca1758a0 | ||
|   | dff3b82f0d | ||
|   | 1b630e7b66 | ||
|   | f4238bf291 | ||
|   | ef8cb8b393 | ||
|   | bed161d485 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 22e0ef4308 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | eb355d110d | ||
|   | c041c295d5 | ||
|   | c582896574 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3e6b59fe1e | ||
|   | 62714b2b68 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 07fdd5b7af | ||
|   | 720f435987 | ||
|   | 52061d6c1a | ||
|   | ae35164a57 | ||
|   | d1c814bd6b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bb50512c89 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0fae45edc9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0a8d3cc8fa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | db09947a67 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5eb600726f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 17a2e6e1f6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 53e7959d54 | ||
|   | 5f140c5fc4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 688b8e5229 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 34b50b45a3 | ||
|   | c17c9c6cc9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c9d72b5253 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f5dbb28fb2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9acfe5c1cc | ||
|   | 701cbcfbad | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 38685127d2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4275f6c6b6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bfc186b612 | ||
|   | 2cbcf1a689 | ||
|   | 1c1c0d70c5 | ||
|   | a66f5fb573 | ||
|   | 9affeab755 | ||
|   | 2bfaf77908 | ||
|   | bc4caae796 | ||
|   | 8746acd329 | ||
|   | 96ecf16da2 | ||
|   | 1e95a0f3ef | ||
|   | a164d793b1 | ||
|   | cb4d92ccf4 | ||
|   | 1dc7256fb5 | ||
|   | 012e710e45 | ||
|   | 5abb7d0286 | ||
|   | ce74946706 | ||
|   | bf351d67e9 | ||
|   | b75fa013d2 | ||
|   | 2601b0d89c | ||
|   | ef8410e121 | ||
|   | 37610703c8 | ||
|   | 4efd9bba8a | ||
|   | e1fe7976d8 | ||
|   | 53b96107d9 | ||
|   | 510fc71b40 | ||
|   | 2a6a3edb77 | ||
|   | c7a8796a47 | ||
|   | dcbad9e798 | ||
|   | 26b3212c7e | ||
|   | 9d40fa5f2b | ||
|   | 8f2a023775 | ||
|   | 989b0b34fe | ||
|   | f3f4bcfe45 | ||
|   | cf94e71215 | ||
|   | 49896f3fa6 | ||
|   | 7cfa9de75f | ||
|   | fc4b7674b1 | ||
|   | 04c9f32539 | ||
|   | 21e3fc9bb9 | ||
|   | 4b78eb7656 | ||
|   | b66dc8894d | ||
|   | 14a7813ab0 | ||
|   | 70a2ca281f | ||
|   | d982f042fc | ||
|   | e60f9e326b | ||
|   | e6f91aef8e | ||
|   | 8f99f86c8b | ||
|   | b7eff547c7 | ||
|   | ba39d189e7 | ||
|   | 78867b2cd9 | ||
|   | ceb6b64152 | ||
|   | d253041376 | ||
|   | cb0aa81f89 | ||
|   | 42061b2f8c | ||
|   | 69bfb89a65 | ||
|   | e0307f9688 | ||
|   | 1cf353461f | ||
|   | 1786235c86 | ||
|   | 645ba3f9c1 | ||
|   | b65f6f46e1 | ||
|   | 84ad521b3d | ||
|   | dfb9c662e7 | ||
|   | 5ac42e17b0 | ||
|   | be2f19637e | ||
|   | 1dff42dc00 | ||
|   | 0c9b3a0765 | ||
|   | 5a109c0ba8 | ||
|   | f3b214c30a | ||
|   | c49d2a0be6 | ||
|   | c6c3170c1b | ||
|   | 0abb958aea | ||
|   | 9d55843629 | ||
|   | b70d309297 | ||
|   | 5961b71562 | ||
|   | 6942626f60 | ||
|   | 069c0acdff | ||
|   | 1f0d83190d | ||
|   | b7a6ee3792 | ||
|   | 1fb2f0c989 | ||
|   | 7c6c92c856 | ||
|   | b4ad411e6f | ||
|   | 5d76a92f73 | ||
|   | beee09491a | ||
|   | ee5aabdddf | ||
|   | ec80f6a6f1 | ||
|   | 9845f0b47c | ||
|   | cd294ba619 | ||
|   | 61e27cb1ea | ||
|   | 8d6295e8e8 | ||
|   | b0e95699f7 | ||
|   | c8e1e7b8a8 | ||
|   | d2cea159af | ||
|   | eb5d1c79c8 | ||
|   | 65ab6848ab | ||
|   | 7a1d934e8d | ||
|   | cbacde12fa | ||
|   | eff352cde1 | ||
|   | 62a75c188c | ||
|   | 4ffa6b6186 | ||
|   | 25173cf605 | ||
|   | 3277d8e80b | ||
|   | 55864fdc82 | ||
|   | d4d662ba46 | ||
|   | 3ea5f508bb | ||
|   | 902a5dd678 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4a3ed62583 | ||
|   | b4223e9e92 | ||
|   | 99955d7818 | ||
|   | f66768726c | ||
|   | 0e4be02b2c | ||
|   | 6daea23b3c | ||
|   | e21ddcb1e5 | ||
|   | ded85d9f27 | ||
|   | eea43494da | ||
|   | 9cf9ef927d | ||
|   | 779ec4f583 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c541831cd2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fd20c2a554 | ||
|   | 14fd29808c | ||
|   | 7132ee157f | ||
|   | 1596b313d5 | ||
|   | 4c33618e05 | ||
|   | 3837b3e630 | ||
|   | 7c15633f6d | ||
|   | f7ec8650eb | ||
|   | 7674eee0fb | ||
|   | f494a6453a | ||
|   | 37f3682ffa | ||
|   | 8055286a1f | ||
|   | 70cd68ded7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cc91a6185e | ||
|   | 1fd7c84583 | ||
|   | 0269540ee9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 98390b3843 | ||
|   | 269628929c | ||
|   | 21fcc84afd | ||
|   | b86c1db83d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a376670478 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 72c62079aa | ||
|   | 9baf875585 | ||
|   | 175915218f | ||
|   | 0bdd213761 | ||
|   | 810b43760e | ||
|   | 424d71c55a | ||
|   | 176924241c | ||
|   | da08aa7fb0 | ||
|   | 6047227648 | ||
|   | fc71fd6bc3 | ||
|   | 90a1b135e1 | ||
|   | e19413b6ca | ||
|   | 0dfc10af5f | ||
|   | bbbc419bea | ||
|   | 50ad5e376f | ||
|   | a9f2254bbc | ||
|   | a8836404d4 | ||
|   | 25f25243bd | ||
|   | cf8d36b1f3 | ||
|   | e3a9d754df | ||
|   | 7b303a699b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ee45eb00f7 | ||
|   | 24a6aa2669 | ||
|   | 66d011cfb9 | ||
|   | 35895735cc | ||
|   | e71df0b71a | ||
|   | 2a9846c598 | ||
|   | b243d56bee | ||
|   | 6a372a165e | ||
|   | a5dad9bc22 | ||
|   | 954e0a5f63 | ||
|   | 4dbd4eebaa | ||
|   | 09b01df366 | ||
|   | a76539c732 | ||
|   | c7babe884c | ||
|   | ce83feec93 | ||
|   | 150ee3fb12 | ||
|   | 8fd3fcd323 | ||
|   | 6e3b3a53e4 | ||
|   | 22966485c7 | ||
|   | 673ca8ba4b | ||
|   | c8be25dfc2 | ||
|   | edaaa00038 | ||
|   | 2de605d97a | ||
|   | 0b11302b1d | ||
|   | ddb224e145 | ||
|   | 317149e51e | ||
|   | 51840b88b3 | ||
|   | 319a1ad8c6 | ||
|   | d75c84750d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 492a73e345 | ||
|   | 64bf101c95 | ||
|   | 876ced25f1 | ||
|   | 72e3b72854 | ||
|   | 6ccd3d3b95 | ||
|   | 5709cb6aa4 | ||
|   | 1fe7282b0e | ||
|   | 6d29063b35 | ||
|   | d3e0b94d27 | ||
|   | f4f1f98433 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 69e3d8e13d | ||
|   | 3ab6e389d3 | ||
|   | 9cc85bc928 | ||
|   | 4f4343d6c8 | ||
|   | 7ab27d620a | ||
|   | fa4ee71803 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 82026250c5 | ||
|   | bc7533bb42 | ||
|   | 7110c0381f | ||
|   | 87fa05accc | ||
|   | 56ee3f82fb | ||
|   | 11872b076b | ||
|   | 620db4238d | ||
|   | ef78bec48d | ||
|   | 4dcf2287ce | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a9766ed66b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 26a83feeb0 | ||
|   | 8c5dd7cdba | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0d025a2355 | ||
|   | 8216778d0c | ||
|   | fe16b689a8 | ||
|   | 2653f6c874 | ||
|   | 8093f7f4cb | ||
|   | b26c914ff9 | ||
|   | b2fa97b6dc | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 8a8bba422a | ||
|   | 76ca66b1b5 | ||
|   | 280dbfc958 | ||
|   | 0b10ad3e78 | ||
|   | c172f0c486 | ||
|   | 1244ed73a2 | ||
|   | e2aef205cc | ||
|   | 7434c9345a | ||
|   | 287ff17107 | ||
|   | 21309944e5 | ||
|   | c0e39ffd67 | ||
|   | 3da2cb3123 | ||
|   | 7957bd1f25 | ||
|   | 04d0aa2f22 | ||
|   | 4b901101da | ||
|   | 4960284e2d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 11d32300e9 | ||
|   | f131e93e63 | ||
|   | 5a540dd889 | ||
|   | 3a70310f78 | ||
|   | feed58c33e | ||
|   | e5585e13fe | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | edd49e1511 | ||
|   | 057fad55e8 | ||
|   | d1a8165de2 | ||
|   | 31c555445d | ||
|   | 61bb5d33e5 | ||
|   | 20a3bab5bc | ||
|   | e8d916acd7 | ||
|   | 8b73d664b4 | ||
|   | 10dcc08068 | ||
|   | fc5adc3753 | ||
|   | d1db8f456f | ||
|   | 0dd07a395a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 589771df5c | ||
|   | 92812048dc | ||
|   | 9fc14d6627 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3da6b85593 | ||
|   | d2a4f481be | ||
|   | e37f67c548 | ||
|   | e775a6770b | ||
|   | 4ba5ef6c37 | ||
|   | d528ab06d9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 03a628cfe2 | ||
|   | 973851b332 | ||
|   | 4c5795c276 | ||
|   | 41d016d96a | ||
|   | 22e647cad4 | ||
|   | b63dd9dbbf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3b3b9e269d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d2c3b9ee83 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5b50a8692b | ||
|   | 8a8bbee8e0 | ||
|   | 28e28d1417 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ea77a0f3d6 | ||
|   | 10e09b238a | ||
|   | f9cd2b66cb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b1c0fba8cf | ||
|   | fdae6257b3 | ||
|   | 6a48aea128 | ||
|   | a90b173671 | ||
|   | d9971bfaa9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 369d56a809 | ||
|   | 939a3cdf63 | ||
|   | 208fd0662c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f133f246cb | ||
|   | b9b8997d68 | ||
|   | 46c4a19a13 | ||
|   | 8d63654211 | ||
|   | 3bf25f125b | ||
|   | 8c65876413 | ||
|   | 2ab6d49553 | ||
|   | 67b0cf0952 | ||
|   | 5138276f8a | ||
|   | 30e6777529 | ||
|   | 1686ab4b9d | ||
|   | b7102c0d7d | ||
|   | d41d524850 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4f05f6305a | ||
|   | ba0b1239be | ||
|   | 708b68f35d | ||
|   | 3108e98b97 | ||
|   | ba7609cc2c | ||
|   | 506fd7d480 | ||
|   | 9767ebe1fb | ||
|   | 539e89e7b5 | ||
|   | a7eef81272 | ||
|   | 7986be103f | ||
|   | 055e65c45e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fe762e9ae4 | ||
|   | 5267c6fdfc | ||
|   | 8eff913845 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1c845d0052 | ||
|   | 60a1d25e1e | ||
|   | 3439d1d663 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bf120d9cb2 | ||
|   | b5a024c879 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 602d754e5e | ||
|   | b7c4f4029d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7fdb5d4862 | ||
|   | bc52ab410c | ||
|   | 3b0220fa92 | ||
|   | 64e00e559f | ||
|   | b407bd4c4f | ||
|   | d466abf9c4 | ||
|   | 4d98230145 | ||
|   | 8a5bca0eb0 | ||
|   | 1638da858c | ||
|   | bfb11102cc | ||
|   | a3d3539e82 | ||
|   | 1fc6cff857 | ||
|   | c5d7eb5384 | ||
|   | 759e6eba35 | ||
|   | 153129e066 | ||
|   | 7bf3c7273e | ||
|   | 08765e6ce2 | ||
|   | a60c9f788d | ||
|   | d9c297c06a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3789bebb2b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bbecf5f368 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e580b30219 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ed8c8ad3e3 | ||
|   | 4f61d5689b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 60a18185d7 | ||
|   | e0246b8488 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1cd0fae84a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e8a1ebbff4 | ||
|   | c5010b8502 | ||
|   | a7db401b62 | ||
|   | 49c7dad6eb | ||
|   | 521c3d40b7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 709a1d2ef0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3c5d7b97d1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9165c8bc57 | ||
|   | 0b3e4eab23 | ||
|   | 39d14c943c | ||
|   | 09469be93f | ||
|   | 6e215870ef | ||
|   | d5985dcaaf | ||
|   | bbd9d8887d | ||
|   | 9588987e30 | ||
|   | 52c05a4426 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e8224df4e5 | ||
|   | 83a6df1621 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c46ebc8d3e | ||
|   | fca530411f | ||
|   | c2c64b9923 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 9968c27a8e | ||
|   | 96796ac5da | ||
|   | 550e4cd4aa | ||
|   | f6041c5cbb | ||
|   | 42f65c2ca1 | ||
|   | 588f171f7f | ||
|   | bd1445840d | ||
|   | 37def6d3e4 | ||
|   | 013d603ba0 | ||
|   | b76407d28d | ||
|   | 4e969ccf09 | ||
|   | 97a0903cec | ||
|   | e2525a3d07 | ||
|   | 8bc0f5a42c | ||
|   | bf7e8ffd24 | ||
|   | 255e598c65 | ||
|   | 5e22178225 | ||
|   | 56b7a6abec | ||
|   | f34c4a11af | ||
|   | 102689b711 | ||
|   | b16d769192 | ||
|   | 29bcacc64e | ||
|   | c7f3331373 | ||
|   | c32444b70c | ||
|   | 11c6b90eb0 | ||
|   | f7a17598f0 | ||
|   | 56967bc0c1 | ||
|   | cdfd6431c3 | ||
|   | c363995718 | ||
|   | 53497aa632 | ||
|   | 8d89b0e57f | ||
|   | 92cf8b5579 | ||
|   | 6068c32176 | ||
|   | 38893324af | ||
|   | a39ab3c174 | ||
|   | 797d2be5bf | ||
|   | 99a91e1019 | ||
|   | 5de8d07ce0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3a31a4a721 | ||
|   | 05f4419a92 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5ea8feb86b | ||
|   | 8fd70b3ae6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 343aa40bc8 | ||
|   | 6022f9a77e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bd9de0680e | ||
|   | b8000d5bc1 | ||
|   | c6efa1127f | ||
|   | 688a3d91d3 | ||
|   | 68151a2a70 | ||
|   | b01ab9234b | ||
|   | ad39228dea | ||
|   | 8cc48cdecb | ||
|   | 524e89acf0 | ||
|   | 48f6b34882 | ||
|   | 44d9185574 | ||
|   | 51ff6c6564 | ||
|   | b49b8e3db8 | ||
|   | c2ca556151 | ||
|   | df86b27af4 | ||
|   | eba1f401cc | ||
|   | 19c2f9c9e8 | ||
|   | 4250447d14 | ||
|   | 4666197f28 | ||
|   | a5ca36c93f | ||
|   | a88950e16c | ||
|   | c013c5ec64 | ||
|   | 53d5d0efbd | ||
|   | 3577991553 | ||
|   | fa758f2bee | ||
|   | 6dbfc2f4ed | ||
|   | b355556c07 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fec336260e | ||
|   | 3e67d91d1a | ||
|   | 641e406502 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 073ba22233 | ||
|   | 6b4a4e6024 | ||
|   | 7d8b418a81 | ||
|   | c14425b2d1 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4740a71bdd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6d0e0158ea | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e966d6f4f4 | ||
|   | b99bb60cd0 | ||
|   | 080c79234c | ||
|   | 9ad887942e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 27dfa514e7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ab5c5389e8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 219679bce9 | ||
|   | aca4a1f86d | ||
|   | f428d6b3f2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 109c3e86d9 | ||
|   | e4b6c3fd4d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 43f1d9be44 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5c405201b2 | ||
|   | 9fc91fbbcc | ||
|   | d9e3f2c15f | ||
|   | 3e2f5b0dd3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | aae1a3604c | ||
|   | e31b9c4264 | ||
|   | 90a9dbafbf | ||
|   | 7041557ee2 | ||
|   | 3d6e5ef1f0 | ||
|   | d9bf605c3f | ||
|   | 20dab92ad8 | ||
|   | 98ed3bdd4d | ||
|   | f5bc6309ae | ||
|   | 620ebd8a61 | ||
|   | ca30af5c8a | ||
|   | 9d30ce348f | ||
|   | 07c7b07362 | ||
|   | c13a80ce5e | ||
|   | 13868478f7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 77aca59dda | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b86605949b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cd19022e2e | ||
|   | 03368c1859 | ||
|   | 8e0ed288e1 | ||
|   | 879e0ed3d5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 657275fd17 | ||
|   | 3d1c908a01 | ||
|   | 713e8e7b71 | ||
|   | 9e597d22a5 | ||
|   | 259e8a14da | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4de4243b55 | ||
|   | a667cb627b | ||
|   | 9461634670 | ||
|   | 51b79b33fb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 539884295b | ||
|   | a885c7e358 | ||
|   | fa968f49c1 | ||
|   | cf3c40f5f7 | ||
|   | 361474885f | ||
|   | 1e06046bd6 | ||
|   | 4c940e62f3 | ||
|   | 7631c409e1 | ||
|   | e1192403d9 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f5c49c83a0 | ||
|   | faae7a2322 | ||
|   | f6aa55ef74 | ||
|   | 1a0afc5079 | ||
|   | 4a50ca4ea5 | ||
|   | 6cb27ede09 | ||
|   | 1b68c51a05 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5f2b11ca9f | ||
|   | ec6666a4ea | ||
|   | a10dbb64f0 | ||
|   | aa17be0e33 | ||
|   | a8919703ee | ||
|   | 1c66a5e437 | ||
|   | 767d785d04 | ||
|   | 0839528e22 | ||
|   | 1012245ef7 | ||
|   | 0d6db8b834 | ||
|   | adea2efb01 | ||
|   | 818914b837 | ||
|   | b207528ecf | ||
|   | 039ef18d8c | ||
|   | db387834f2 | ||
|   | 1b7d9f9e3b | ||
|   | ed8c9f5ce5 | ||
|   | c3bf1d8770 | ||
|   | 7db4693082 | ||
|   | 72a12a4ba4 | ||
|   | aee9e4b0a5 | ||
|   | 55f6affc9e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1e5f2f7215 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | da864d5bb7 | ||
|   | 8d60f39cf4 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2045519814 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3f70e88a4f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6ed5fbe102 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4e72d5083d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | f9e102e537 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e54363875b | ||
|   | 1ce4e2a799 | ||
|   | 80b86a89f0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 1a316d251e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a74f90b768 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | fc6e457581 | ||
|   | ad7b8b66f2 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0714677a8a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 0a946a5c43 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 4930a8d88e | ||
|   | 8b781cec8e | ||
|   | 065c98c5d7 | ||
|   | 6a3c58d20f | ||
|   | a87afe9fb3 | ||
|   | 61fe8983f3 | ||
|   | c10410ade3 | ||
|   | 761fded9e3 | ||
|   | b87fbe7a1e | ||
|   | 7fdf824e97 | ||
|   | 69d8eeb7db | ||
|   | 3b7d2869e5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | bcda5cd0cf | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | eeb64a25ff | ||
|   | 9134132ba9 | ||
|   | 1ded254e5a | ||
|   | fc104a7992 | ||
|   | e7e062a222 | ||
|   | 5233086efb | ||
|   | 8d95f0d95d | ||
|   | 5cf8b39703 | ||
|   | 15dabe372c | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | aab52a8bb2 | ||
|   | aa52825b40 | ||
|   | 2809a306e6 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | a6304d6284 | ||
|   | 8e866e86d6 | ||
|   | 2e8203f666 | ||
|   | b60f2e3201 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | c5f57f436c | ||
|   | 3bb930b906 | ||
|   | e75331e159 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | d6b66a7145 | ||
|   | 5c346798c8 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5ffe37407a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2b056c0434 | ||
|   | 27b36707e5 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 5760614b65 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 3835912b01 | ||
|   | a385655c85 | ||
|   | e177012108 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | cc3234ad8f | ||
|   | 4d932f0b4a | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 257769cdc7 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6619f064eb | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 382a47a082 | ||
|   | fe946eb75b | ||
|   | 3bfbe4bde6 | ||
|   | 7963a97358 | ||
|   | 0ed2b5966e | ||
|   | 70c5f77aa7 | ||
|   | 1013647249 | ||
|   | 45e9c51525 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ad4be75fe1 | ||
|   | d605b67b41 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dba6a3c756 | ||
|   | 002e9ad071 | ||
|   | 6e7874c2c9 | ||
|   | 978f9b0f83 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 2b88669a72 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 252fd2bb6c | ||
|   | cc68a087a2 | ||
|   | bbe3a9e0c2 | ||
|   | 33ea02208a | ||
|   | cf531cd935 | ||
|   | 232649c0cd | ||
|   | 1db8ef37a2 | ||
|   | eecd765d09 | ||
|   | 3d75831623 | ||
|   | c1934e0b9a | ||
|   | c0e9c3b9dc | ||
|   | 3e1341a731 | ||
|   | 6be25270fd | ||
|   | ce929aea46 | ||
|   | 8853bf6ea2 | ||
|   | 2241807745 | ||
|   | 50d705c943 | ||
|   | eb111d3c32 | ||
|   | 1e59f9f4be | ||
|   | 523eb9522f | ||
|   | f6cb322819 | ||
|   | 4f97756f4e | ||
|   | 8644dd5271 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 26d842f432 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ad4f14ffaf | ||
|   | 948c858e78 | ||
|   | 49099223d3 | ||
|   | 0fbd430594 | ||
|   | 8cc762d839 | ||
|   | 89d9dd2893 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b7d1ce1c37 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 869d10ca3f | ||
|   | f338089148 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 06b0f9fcaf | ||
|   | 7ad07e4c55 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ad65600d11 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | e91d907e56 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | b35a1fc9e0 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | dd18ad96f3 | ||
|   | 62eec56e5f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 7187e25cad | ||
|   | 6d9e6a616d | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 44d87e3c66 | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 085e2460bc | ||
|   | 1f8a9e4caf | ||
|   | f08877437e | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 6690d1ef22 | ||
|   | 9d8a5b366e | ||
|   | 22c798c9d6 | ||
|   | 8aabb1f32f | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 33d5cecc85 | ||
|   | 7693a4dc24 | ||
|   | 9ec38c7dd9 | ||
|   | e8cb85f7ff | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ef964a2717 | ||
|   | 369881f8a6 | ||
|   | 68e22d23f1 | ||
|   | 696ba69a9e | ||
|   | e2ab52e10e | ||
|   | b154bc1502 | ||
|   | a952b880d8 | ||
|   | 018aceb542 | ||
|   | 2fb86f118e | ||
|   | 6c8caccfec | ||
|   | 3dd3a80054 | ||
|   | 675310afdf | ||
|   | f5df91d4c7 | ||
|   | d8ab9b73ba | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | 89ab0b4a3d | ||
|   | dd4cb1df72 | ||
|   | b81cd37776 | ||
|   | 34112e7446 | ||
|   | 2dee45b465 | ||
|   | 10eb0a8b87 | ||
|   | 551035238f | ||
|   | eb9359e9e1 | 
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/bug_report.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ body: | |||||||
|  |  | ||||||
|         **Please do not report issues for custom cards.** |         **Please do not report issues for custom cards.** | ||||||
|  |  | ||||||
|         [fr]: https://github.com/home-assistant/frontend/discussions |         [fr]: https://github.com/orgs/home-assistant/discussions | ||||||
|         [releases]: https://github.com/home-assistant/home-assistant/releases |         [releases]: https://github.com/home-assistant/home-assistant/releases | ||||||
|   - type: checkboxes |   - type: checkboxes | ||||||
|     attributes: |     attributes: | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,7 +1,7 @@ | |||||||
| blank_issues_enabled: false | blank_issues_enabled: false | ||||||
| contact_links: | contact_links: | ||||||
|   - name: Request a feature for the UI / Dashboards |   - name: Request a feature for the UI / Dashboards | ||||||
|     url: https://github.com/home-assistant/frontend/discussions/category_choices |     url: https://github.com/orgs/home-assistant/discussions | ||||||
|     about: Request a new feature for the Home Assistant frontend. |     about: Request a new feature for the Home Assistant frontend. | ||||||
|   - name: Report a bug that is NOT related to the UI / Dashboards |   - 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 | ||||||
|   | |||||||
							
								
								
									
										53
									
								
								.github/ISSUE_TEMPLATE/task.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								.github/ISSUE_TEMPLATE/task.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | |||||||
|  | name: Task | ||||||
|  | description: For staff only - Create a task | ||||||
|  | type: Task | ||||||
|  | body: | ||||||
|  |   - type: markdown | ||||||
|  |     attributes: | ||||||
|  |       value: | | ||||||
|  |         ## ⚠️ RESTRICTED ACCESS | ||||||
|  |  | ||||||
|  |         **This form is restricted to Open Home Foundation staff and authorized contributors only.** | ||||||
|  |  | ||||||
|  |         If you are a community member wanting to contribute, please: | ||||||
|  |         - For bug reports: Use the [bug report form](https://github.com/home-assistant/frontend/issues/new?template=bug_report.yml) | ||||||
|  |         - For feature requests: Submit to [Feature Requests](https://github.com/orgs/home-assistant/discussions) | ||||||
|  |  | ||||||
|  |         --- | ||||||
|  |  | ||||||
|  |         ### For authorized contributors | ||||||
|  |  | ||||||
|  |         Use this form to create tasks for development work, improvements, or other actionable items that need to be tracked. | ||||||
|  |   - type: textarea | ||||||
|  |     id: description | ||||||
|  |     attributes: | ||||||
|  |       label: Description | ||||||
|  |       description: | | ||||||
|  |         Provide a clear and detailed description of the task that needs to be accomplished. | ||||||
|  |  | ||||||
|  |         Be specific about what needs to be done, why it's important, and any constraints or requirements. | ||||||
|  |       placeholder: | | ||||||
|  |         Describe the task, including: | ||||||
|  |         - What needs to be done | ||||||
|  |         - Why this task is needed | ||||||
|  |         - Expected outcome | ||||||
|  |         - Any constraints or requirements | ||||||
|  |     validations: | ||||||
|  |       required: true | ||||||
|  |   - type: textarea | ||||||
|  |     id: additional_context | ||||||
|  |     attributes: | ||||||
|  |       label: Additional context | ||||||
|  |       description: | | ||||||
|  |         Any additional information, links, research, or context that would be helpful. | ||||||
|  |  | ||||||
|  |         Include links to related issues, research, prototypes, roadmap opportunities etc. | ||||||
|  |       placeholder: | | ||||||
|  |         - Roadmap opportunity: [link] | ||||||
|  |         - Epic: [link] | ||||||
|  |         - Feature request: [link] | ||||||
|  |         - Technical design documents: [link] | ||||||
|  |         - Prototype/mockup: [link] | ||||||
|  |         - Dependencies: [links] | ||||||
|  |     validations: | ||||||
|  |       required: false | ||||||
							
								
								
									
										596
									
								
								.github/copilot-instructions.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										596
									
								
								.github/copilot-instructions.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,596 @@ | |||||||
|  | # GitHub Copilot & Claude Code Instructions | ||||||
|  |  | ||||||
|  | You are an assistant helping with development of the Home Assistant frontend. The frontend is built using Lit-based Web Components and TypeScript, providing a responsive and performant interface for home automation control. | ||||||
|  |  | ||||||
|  | ## Table of Contents | ||||||
|  |  | ||||||
|  | - [Quick Reference](#quick-reference) | ||||||
|  | - [Core Architecture](#core-architecture) | ||||||
|  | - [Development Standards](#development-standards) | ||||||
|  | - [Component Library](#component-library) | ||||||
|  | - [Common Patterns](#common-patterns) | ||||||
|  | - [Text and Copy Guidelines](#text-and-copy-guidelines) | ||||||
|  | - [Development Workflow](#development-workflow) | ||||||
|  | - [Review Guidelines](#review-guidelines) | ||||||
|  |  | ||||||
|  | ## Quick Reference | ||||||
|  |  | ||||||
|  | ### Essential Commands | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | yarn lint          # ESLint + Prettier + TypeScript + Lit | ||||||
|  | yarn format        # Auto-fix ESLint + Prettier | ||||||
|  | yarn lint:types    # TypeScript compiler | ||||||
|  | yarn test          # Vitest | ||||||
|  | script/develop     # Development server | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Component Prefixes | ||||||
|  |  | ||||||
|  | - `ha-` - Home Assistant components | ||||||
|  | - `hui-` - Lovelace UI components | ||||||
|  | - `dialog-` - Dialog components | ||||||
|  |  | ||||||
|  | ### Import Patterns | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | import type { HomeAssistant } from "../types"; | ||||||
|  | import { fireEvent } from "../common/dom/fire_event"; | ||||||
|  | import { showAlertDialog } from "../dialogs/generic/show-alert-dialog"; | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Core Architecture | ||||||
|  |  | ||||||
|  | The Home Assistant frontend is a modern web application that: | ||||||
|  |  | ||||||
|  | - Uses Web Components (custom elements) built with Lit framework | ||||||
|  | - Is written entirely in TypeScript with strict type checking | ||||||
|  | - Communicates with the backend via WebSocket API | ||||||
|  | - Provides comprehensive theming and internationalization | ||||||
|  |  | ||||||
|  | ## Development Standards | ||||||
|  |  | ||||||
|  | ### Code Quality Requirements | ||||||
|  |  | ||||||
|  | **Linting and Formatting (Enforced by Tools)** | ||||||
|  |  | ||||||
|  | - ESLint config extends Airbnb, TypeScript strict, Lit, Web Components, Accessibility | ||||||
|  | - Prettier with ES5 trailing commas enforced | ||||||
|  | - No console statements (`no-console: "error"`) - use proper logging | ||||||
|  | - Import organization: No unused imports, consistent type imports | ||||||
|  |  | ||||||
|  | **Naming Conventions** | ||||||
|  |  | ||||||
|  | - PascalCase for types and classes | ||||||
|  | - camelCase for variables, methods | ||||||
|  | - Private methods require leading underscore | ||||||
|  | - Public methods forbid leading underscore | ||||||
|  |  | ||||||
|  | ### TypeScript Usage | ||||||
|  |  | ||||||
|  | - **Always use strict TypeScript**: Enable all strict flags, avoid `any` types | ||||||
|  | - **Proper type imports**: Use `import type` for type-only imports | ||||||
|  | - **Define interfaces**: Create proper interfaces for data structures | ||||||
|  | - **Type component properties**: All Lit properties must be properly typed | ||||||
|  | - **No unused variables**: Prefix with `_` if intentionally unused | ||||||
|  | - **Consistent imports**: Use `@typescript-eslint/consistent-type-imports` | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | // Good | ||||||
|  | import type { HomeAssistant } from "../types"; | ||||||
|  |  | ||||||
|  | interface EntityConfig { | ||||||
|  |   entity: string; | ||||||
|  |   name?: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @property({ type: Object }) | ||||||
|  | hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  | // Bad | ||||||
|  | @property() | ||||||
|  | hass: any; | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Web Components with Lit | ||||||
|  |  | ||||||
|  | - **Use Lit 3.x patterns**: Follow modern Lit practices | ||||||
|  | - **Extend appropriate base classes**: Use `LitElement`, `SubscribeMixin`, or other mixins as needed | ||||||
|  | - **Define custom element names**: Use `ha-` prefix for components | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | @customElement("ha-my-component") | ||||||
|  | export class HaMyComponent extends LitElement { | ||||||
|  |   @property({ attribute: false }) | ||||||
|  |   hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  |   @state() | ||||||
|  |   private _config?: MyComponentConfig; | ||||||
|  |  | ||||||
|  |   static get styles() { | ||||||
|  |     return css` | ||||||
|  |       :host { | ||||||
|  |         display: block; | ||||||
|  |       } | ||||||
|  |     `; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   render() { | ||||||
|  |     return html`<div>Content</div>`; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Component Guidelines | ||||||
|  |  | ||||||
|  | - **Use composition**: Prefer composition over inheritance | ||||||
|  | - **Lazy load panels**: Heavy panels should be dynamically imported | ||||||
|  | - **Optimize renders**: Use `@state()` for internal state, `@property()` for public API | ||||||
|  | - **Handle loading states**: Always show appropriate loading indicators | ||||||
|  | - **Support themes**: Use CSS custom properties from theme | ||||||
|  |  | ||||||
|  | ### Data Management | ||||||
|  |  | ||||||
|  | - **Use WebSocket API**: All backend communication via home-assistant-js-websocket | ||||||
|  | - **Cache appropriately**: Use collections and caching for frequently accessed data | ||||||
|  | - **Handle errors gracefully**: All API calls should have error handling | ||||||
|  | - **Update real-time**: Subscribe to state changes for live updates | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | // Good | ||||||
|  | try { | ||||||
|  |   const result = await fetchEntityRegistry(this.hass.connection); | ||||||
|  |   this._processResult(result); | ||||||
|  | } catch (err) { | ||||||
|  |   showAlertDialog(this, { | ||||||
|  |     text: `Failed to load: ${err.message}`, | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Styling Guidelines | ||||||
|  |  | ||||||
|  | - **Use CSS custom properties**: Leverage the theme system | ||||||
|  | - **Mobile-first responsive**: Design for mobile, enhance for desktop | ||||||
|  | - **Follow Material Design**: Use Material Web Components where appropriate | ||||||
|  | - **Support RTL**: Ensure all layouts work in RTL languages | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | static get styles() { | ||||||
|  |   return css` | ||||||
|  |     :host { | ||||||
|  |       --spacing: 16px; | ||||||
|  |       padding: var(--spacing); | ||||||
|  |       color: var(--primary-text-color); | ||||||
|  |       background-color: var(--card-background-color); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @media (max-width: 600px) { | ||||||
|  |       :host { | ||||||
|  |         --spacing: 8px; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   `; | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Performance Best Practices | ||||||
|  |  | ||||||
|  | - **Code split**: Split code at the panel/dialog level | ||||||
|  | - **Lazy load**: Use dynamic imports for heavy components | ||||||
|  | - **Optimize bundle**: Keep initial bundle size minimal | ||||||
|  | - **Use virtual scrolling**: For long lists, implement virtual scrolling | ||||||
|  | - **Memoize computations**: Cache expensive calculations | ||||||
|  |  | ||||||
|  | ### Testing Requirements | ||||||
|  |  | ||||||
|  | - **Write tests**: Add tests for data processing and utilities | ||||||
|  | - **Test with Vitest**: Use the established test framework | ||||||
|  | - **Mock appropriately**: Mock WebSocket connections and API calls | ||||||
|  | - **Test accessibility**: Ensure components are accessible | ||||||
|  |  | ||||||
|  | ## Component Library | ||||||
|  |  | ||||||
|  | ### Dialog Components | ||||||
|  |  | ||||||
|  | **Available Dialog Types:** | ||||||
|  |  | ||||||
|  | - `ha-md-dialog` - Preferred for new code (Material Design 3) | ||||||
|  | - `ha-dialog` - Legacy component still widely used | ||||||
|  |  | ||||||
|  | **Opening Dialogs (Fire Event Pattern - Recommended):** | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | fireEvent(this, "show-dialog", { | ||||||
|  |   dialogTag: "dialog-example", | ||||||
|  |   dialogImport: () => import("./dialog-example"), | ||||||
|  |   dialogParams: { title: "Example", data: someData }, | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | **Dialog Implementation Requirements:** | ||||||
|  |  | ||||||
|  | - Implement `HassDialog<T>` interface | ||||||
|  | - Use `createCloseHeading()` for standard headers | ||||||
|  | - Import `haStyleDialog` for consistent styling | ||||||
|  | - Return `nothing` when no params (loading state) | ||||||
|  | - Fire `dialog-closed` event when closing | ||||||
|  | - Add `dialogInitialFocus` for accessibility | ||||||
|  |  | ||||||
|  | ```` | ||||||
|  |  | ||||||
|  | ### Form Component (ha-form) | ||||||
|  | - Schema-driven using `HaFormSchema[]` | ||||||
|  | - Supports entity, device, area, target, number, boolean, time, action, text, object, select, icon, media, location selectors | ||||||
|  | - Built-in validation with error display | ||||||
|  | - Use `dialogInitialFocus` in dialogs | ||||||
|  | - Use `computeLabel`, `computeError`, `computeHelper` for translations | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | <ha-form | ||||||
|  |   .hass=${this.hass} | ||||||
|  |   .data=${this._data} | ||||||
|  |   .schema=${this._schema} | ||||||
|  |   .error=${this._errors} | ||||||
|  |   .computeLabel=${(schema) => this.hass.localize(`ui.panel.${schema.name}`)} | ||||||
|  |   @value-changed=${this._valueChanged} | ||||||
|  | ></ha-form> | ||||||
|  | ```` | ||||||
|  |  | ||||||
|  | ### Alert Component (ha-alert) | ||||||
|  |  | ||||||
|  | - Types: `error`, `warning`, `info`, `success` | ||||||
|  | - Properties: `title`, `alert-type`, `dismissable`, `icon`, `action`, `rtl` | ||||||
|  | - Content announced by screen readers when dynamically displayed | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <ha-alert alert-type="error">Error message</ha-alert> | ||||||
|  | <ha-alert alert-type="warning" title="Warning">Description</ha-alert> | ||||||
|  | <ha-alert alert-type="success" dismissable>Success message</ha-alert> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## Common Patterns | ||||||
|  |  | ||||||
|  | ### Creating a Panel | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | @customElement("ha-panel-myfeature") | ||||||
|  | export class HaPanelMyFeature extends SubscribeMixin(LitElement) { | ||||||
|  |   @property({ attribute: false }) | ||||||
|  |   hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  |   @property({ type: Boolean, reflect: true }) | ||||||
|  |   narrow!: boolean; | ||||||
|  |  | ||||||
|  |   @property() | ||||||
|  |   route!: Route; | ||||||
|  |  | ||||||
|  |   hassSubscribe() { | ||||||
|  |     return [ | ||||||
|  |       subscribeEntityRegistry(this.hass.connection, (entities) => { | ||||||
|  |         this._entities = entities; | ||||||
|  |       }), | ||||||
|  |     ]; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Creating a Dialog | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | @customElement("dialog-my-feature") | ||||||
|  | export class DialogMyFeature | ||||||
|  |   extends LitElement | ||||||
|  |   implements HassDialog<MyDialogParams> | ||||||
|  | { | ||||||
|  |   @property({ attribute: false }) | ||||||
|  |   hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  |   @state() | ||||||
|  |   private _params?: MyDialogParams; | ||||||
|  |  | ||||||
|  |   public async showDialog(params: MyDialogParams): Promise<void> { | ||||||
|  |     this._params = params; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public closeDialog(): void { | ||||||
|  |     this._params = undefined; | ||||||
|  |     fireEvent(this, "dialog-closed", { dialog: this.localName }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   protected render() { | ||||||
|  |     if (!this._params) { | ||||||
|  |       return nothing; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return html` | ||||||
|  |       <ha-dialog | ||||||
|  |         open | ||||||
|  |         @closed=${this.closeDialog} | ||||||
|  |         .heading=${createCloseHeading(this.hass, this._params.title)} | ||||||
|  |       > | ||||||
|  |         <!-- Dialog content --> | ||||||
|  |         <ha-button | ||||||
|  |           appearance="plain" | ||||||
|  |           @click=${this.closeDialog} | ||||||
|  |           slot="secondaryAction" | ||||||
|  |         > | ||||||
|  |           ${this.hass.localize("ui.common.cancel")} | ||||||
|  |         </ha-button> | ||||||
|  |         <ha-button @click=${this._submit} slot="primaryAction"> | ||||||
|  |           ${this.hass.localize("ui.common.save")} | ||||||
|  |         </ha-button> | ||||||
|  |       </ha-dialog> | ||||||
|  |     `; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static styles = [haStyleDialog, css``]; | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Dialog Design Guidelines | ||||||
|  |  | ||||||
|  | - Max width: 560px (Alert/confirmation: 320px fixed width) | ||||||
|  | - Close X-icon on top left (all screen sizes) | ||||||
|  | - Submit button grouped with cancel at bottom right | ||||||
|  | - Keep button labels short: "Save", "Delete", "Enable" | ||||||
|  | - Destructive actions use red warning button | ||||||
|  | - Always use a title (best practice) | ||||||
|  | - Strive for minimalism | ||||||
|  |  | ||||||
|  | #### Creating a Lovelace Card | ||||||
|  |  | ||||||
|  | **Purpose**: Cards allow users to tell different stories about their house (based on gallery) | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | @customElement("hui-my-card") | ||||||
|  | export class HuiMyCard extends LitElement implements LovelaceCard { | ||||||
|  |   @property({ attribute: false }) | ||||||
|  |   hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  |   @state() | ||||||
|  |   private _config?: MyCardConfig; | ||||||
|  |  | ||||||
|  |   public setConfig(config: MyCardConfig): void { | ||||||
|  |     if (!config.entity) { | ||||||
|  |       throw new Error("Entity required"); | ||||||
|  |     } | ||||||
|  |     this._config = config; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public getCardSize(): number { | ||||||
|  |     return 3; // Height in grid units | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Optional: Editor for card configuration | ||||||
|  |   public static getConfigElement(): LovelaceCardEditor { | ||||||
|  |     return document.createElement("hui-my-card-editor"); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Optional: Stub config for card picker | ||||||
|  |   public static getStubConfig(): object { | ||||||
|  |     return { entity: "" }; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | **Card Guidelines:** | ||||||
|  |  | ||||||
|  | - Cards are highly customizable for different households | ||||||
|  | - Implement `LovelaceCard` interface with `setConfig()` and `getCardSize()` | ||||||
|  | - Use proper error handling in `setConfig()` | ||||||
|  | - Consider all possible states (loading, error, unavailable) | ||||||
|  | - Support different entity types and states | ||||||
|  | - Follow responsive design principles | ||||||
|  | - Add configuration editor when needed | ||||||
|  |  | ||||||
|  | ### Internationalization | ||||||
|  |  | ||||||
|  | - **Use localize**: Always use the localization system | ||||||
|  | - **Add translation keys**: Add keys to src/translations/en.json | ||||||
|  | - **Support placeholders**: Use proper placeholder syntax | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | this.hass.localize("ui.panel.config.updates.update_available", { | ||||||
|  |   count: 5, | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Accessibility | ||||||
|  |  | ||||||
|  | - **ARIA labels**: Add appropriate ARIA labels | ||||||
|  | - **Keyboard navigation**: Ensure all interactions work with keyboard | ||||||
|  | - **Screen reader support**: Test with screen readers | ||||||
|  | - **Color contrast**: Meet WCAG AA standards | ||||||
|  |  | ||||||
|  | ## Development Workflow | ||||||
|  |  | ||||||
|  | ### Setup and Commands | ||||||
|  |  | ||||||
|  | 1. **Setup**: `script/setup` - Install dependencies | ||||||
|  | 2. **Develop**: `script/develop` - Development server | ||||||
|  | 3. **Lint**: `yarn lint` - Run all linting before committing | ||||||
|  | 4. **Test**: `yarn test` - Add and run tests | ||||||
|  | 5. **Build**: `script/build_frontend` - Test production build | ||||||
|  |  | ||||||
|  | ### Common Pitfalls to Avoid | ||||||
|  |  | ||||||
|  | - Don't use `querySelector` - Use refs or component properties | ||||||
|  | - Don't manipulate DOM directly - Let Lit handle rendering | ||||||
|  | - Don't use global styles - Scope styles to components | ||||||
|  | - Don't block the main thread - Use web workers for heavy computation | ||||||
|  | - Don't ignore TypeScript errors - Fix all type issues | ||||||
|  |  | ||||||
|  | ### Security Best Practices | ||||||
|  |  | ||||||
|  | - Sanitize HTML - Never use `unsafeHTML` with user content | ||||||
|  | - Validate inputs - Always validate user inputs | ||||||
|  | - Use HTTPS - All external resources must use HTTPS | ||||||
|  | - CSP compliance - Ensure code works with Content Security Policy | ||||||
|  |  | ||||||
|  | ### Text and Copy Guidelines | ||||||
|  |  | ||||||
|  | #### Terminology Standards | ||||||
|  |  | ||||||
|  | **Delete vs Remove** (Based on gallery/src/pages/Text/remove-delete-add-create.markdown) | ||||||
|  |  | ||||||
|  | - **Use "Remove"** for actions that can be restored or reapplied: | ||||||
|  |   - Removing a user's permission | ||||||
|  |   - Removing a user from a group | ||||||
|  |   - Removing links between items | ||||||
|  |   - Removing a widget from dashboard | ||||||
|  |   - Removing an item from a cart | ||||||
|  | - **Use "Delete"** for permanent, non-recoverable actions: | ||||||
|  |   - Deleting a field | ||||||
|  |   - Deleting a value in a field | ||||||
|  |   - Deleting a task | ||||||
|  |   - Deleting a group | ||||||
|  |   - Deleting a permission | ||||||
|  |   - Deleting a calendar event | ||||||
|  |  | ||||||
|  | **Create vs Add** (Create pairs with Delete, Add pairs with Remove) | ||||||
|  |  | ||||||
|  | - **Use "Add"** for already-existing items: | ||||||
|  |   - Adding a permission to a user | ||||||
|  |   - Adding a user to a group | ||||||
|  |   - Adding links between items | ||||||
|  |   - Adding a widget to dashboard | ||||||
|  |   - Adding an item to a cart | ||||||
|  | - **Use "Create"** for something made from scratch: | ||||||
|  |   - Creating a new field | ||||||
|  |   - Creating a new task | ||||||
|  |   - Creating a new group | ||||||
|  |   - Creating a new permission | ||||||
|  |   - Creating a new calendar event | ||||||
|  |  | ||||||
|  | #### Writing Style (Consistent with Home Assistant Documentation) | ||||||
|  |  | ||||||
|  | - **Use American English**: Standard spelling and terminology | ||||||
|  | - **Friendly, informational tone**: Be inspiring, personal, comforting, engaging | ||||||
|  | - **Address users directly**: Use "you" and "your" | ||||||
|  | - **Be inclusive**: Objective, non-discriminatory language | ||||||
|  | - **Be concise**: Use clear, direct language | ||||||
|  | - **Be consistent**: Follow established terminology patterns | ||||||
|  | - **Use active voice**: "Delete the automation" not "The automation should be deleted" | ||||||
|  | - **Avoid jargon**: Use terms familiar to home automation users | ||||||
|  |  | ||||||
|  | #### Language Standards | ||||||
|  |  | ||||||
|  | - **Always use "Home Assistant"** in full, never "HA" or "HASS" | ||||||
|  | - **Avoid abbreviations**: Spell out terms when possible | ||||||
|  | - **Use sentence case everywhere**: Titles, headings, buttons, labels, UI elements | ||||||
|  |   - ✅ "Create new automation" | ||||||
|  |   - ❌ "Create New Automation" | ||||||
|  |   - ✅ "Device settings" | ||||||
|  |   - ❌ "Device Settings" | ||||||
|  | - **Oxford comma**: Use in lists (item 1, item 2, and item 3) | ||||||
|  | - **Replace Latin terms**: Use "like" instead of "e.g.", "for example" instead of "i.e." | ||||||
|  | - **Avoid CAPS for emphasis**: Use bold or italics instead | ||||||
|  | - **Write for all skill levels**: Both technical and non-technical users | ||||||
|  |  | ||||||
|  | #### Key Terminology | ||||||
|  |  | ||||||
|  | - **"add-on"** (hyphenated, not "addon") | ||||||
|  | - **"integration"** (preferred over "component") | ||||||
|  | - **Technical terms**: Use lowercase (automation, entity, device, service) | ||||||
|  |  | ||||||
|  | #### Translation Considerations | ||||||
|  |  | ||||||
|  | - **Add translation keys**: All user-facing text must be translatable | ||||||
|  | - **Use placeholders**: Support dynamic content in translations | ||||||
|  | - **Keep context**: Provide enough context for translators | ||||||
|  |  | ||||||
|  | ```typescript | ||||||
|  | // Good | ||||||
|  | this.hass.localize("ui.panel.config.automation.delete_confirm", { | ||||||
|  |   name: automation.alias, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | // Bad - hardcoded text | ||||||
|  | ("Are you sure you want to delete this automation?"); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Common Review Issues (From PR Analysis) | ||||||
|  |  | ||||||
|  | #### User Experience and Accessibility | ||||||
|  |  | ||||||
|  | - **Form validation**: Always provide proper field labels and validation feedback | ||||||
|  | - **Form accessibility**: Prevent password managers from incorrectly identifying fields | ||||||
|  | - **Loading states**: Show clear progress indicators during async operations | ||||||
|  | - **Error handling**: Display meaningful error messages when operations fail | ||||||
|  | - **Mobile responsiveness**: Ensure components work well on small screens | ||||||
|  | - **Hit targets**: Make clickable areas large enough for touch interaction | ||||||
|  | - **Visual feedback**: Provide clear indication of interactive states | ||||||
|  |  | ||||||
|  | #### Dialog and Modal Patterns | ||||||
|  |  | ||||||
|  | - **Dialog width constraints**: Respect minimum and maximum width requirements | ||||||
|  | - **Interview progress**: Show clear progress for multi-step operations | ||||||
|  | - **State persistence**: Handle dialog state properly during background operations | ||||||
|  | - **Cancel behavior**: Ensure cancel/close buttons work consistently | ||||||
|  | - **Form prefilling**: Use smart defaults but allow user override | ||||||
|  |  | ||||||
|  | #### Component Design Patterns | ||||||
|  |  | ||||||
|  | - **Terminology consistency**: Use "Join"/"Apply" instead of "Group" when appropriate | ||||||
|  | - **Visual hierarchy**: Ensure proper font sizes and spacing ratios | ||||||
|  | - **Grid alignment**: Components should align to the design grid system | ||||||
|  | - **Badge placement**: Position badges and indicators consistently | ||||||
|  | - **Color theming**: Respect theme variables and design system colors | ||||||
|  |  | ||||||
|  | #### Code Quality Issues | ||||||
|  |  | ||||||
|  | - **Null checking**: Always check if entities exist before accessing properties | ||||||
|  | - **TypeScript safety**: Handle potentially undefined array/object access | ||||||
|  | - **Import organization**: Remove unused imports and use proper type imports | ||||||
|  | - **Event handling**: Properly subscribe and unsubscribe from events | ||||||
|  | - **Memory leaks**: Clean up subscriptions and event listeners | ||||||
|  |  | ||||||
|  | #### Configuration and Props | ||||||
|  |  | ||||||
|  | - **Optional parameters**: Make configuration fields optional when sensible | ||||||
|  | - **Smart defaults**: Provide reasonable default values | ||||||
|  | - **Future extensibility**: Design APIs that can be extended later | ||||||
|  | - **Validation**: Validate configuration before applying changes | ||||||
|  |  | ||||||
|  | ## Review Guidelines | ||||||
|  |  | ||||||
|  | ### Core Requirements Checklist | ||||||
|  |  | ||||||
|  | - [ ] TypeScript strict mode passes (`yarn lint:types`) | ||||||
|  | - [ ] No ESLint errors or warnings (`yarn lint:eslint`) | ||||||
|  | - [ ] Prettier formatting applied (`yarn lint:prettier`) | ||||||
|  | - [ ] Lit analyzer passes (`yarn lint:lit`) | ||||||
|  | - [ ] Component follows Lit best practices | ||||||
|  | - [ ] Proper error handling implemented | ||||||
|  | - [ ] Loading states handled | ||||||
|  | - [ ] Mobile responsive | ||||||
|  | - [ ] Theme variables used | ||||||
|  | - [ ] Translations added | ||||||
|  | - [ ] Accessible to screen readers | ||||||
|  | - [ ] Tests added (where applicable) | ||||||
|  | - [ ] No console statements (use proper logging) | ||||||
|  | - [ ] Unused imports removed | ||||||
|  | - [ ] Proper naming conventions | ||||||
|  |  | ||||||
|  | ### Text and Copy Checklist | ||||||
|  |  | ||||||
|  | - [ ] Follows terminology guidelines (Delete vs Remove, Create vs Add) | ||||||
|  | - [ ] Localization keys added for all user-facing text | ||||||
|  | - [ ] Uses "Home Assistant" (never "HA" or "HASS") | ||||||
|  | - [ ] Sentence case for ALL text (titles, headings, buttons, labels) | ||||||
|  | - [ ] American English spelling | ||||||
|  | - [ ] Friendly, informational tone | ||||||
|  | - [ ] Avoids abbreviations and jargon | ||||||
|  | - [ ] Correct terminology (add-on not addon, integration not component) | ||||||
|  |  | ||||||
|  | ### Component-Specific Checks | ||||||
|  |  | ||||||
|  | - [ ] Dialogs implement HassDialog interface | ||||||
|  | - [ ] Dialog styling uses haStyleDialog | ||||||
|  | - [ ] Dialog accessibility includes dialogInitialFocus | ||||||
|  | - [ ] ha-alert used correctly for messages | ||||||
|  | - [ ] ha-form uses proper schema structure | ||||||
|  | - [ ] Components handle all states (loading, error, unavailable) | ||||||
|  | - [ ] Entity existence checked before property access | ||||||
|  | - [ ] Event subscriptions properly cleaned up | ||||||
							
								
								
									
										12
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/cast_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -21,12 +21,12 @@ jobs: | |||||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} |       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           ref: dev |           ref: dev | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -42,7 +42,7 @@ jobs: | |||||||
|       - name: Deploy to Netlify |       - name: Deploy to Netlify | ||||||
|         id: deploy |         id: deploy | ||||||
|         run: | |         run: | | ||||||
|           npx -y netlify-cli deploy --dir=cast/dist --alias dev |           npx -y netlify-cli@23.7.3 deploy --dir=cast/dist --alias dev | ||||||
|         env: |         env: | ||||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} |           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} |           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} | ||||||
| @@ -56,12 +56,12 @@ jobs: | |||||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} |       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           ref: master |           ref: master | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -77,7 +77,7 @@ jobs: | |||||||
|       - name: Deploy to Netlify |       - name: Deploy to Netlify | ||||||
|         id: deploy |         id: deploy | ||||||
|         run: | |         run: | | ||||||
|           npx -y netlify-cli deploy --dir=cast/dist --prod |           npx -y netlify-cli@23.7.3 deploy --dir=cast/dist --prod | ||||||
|         env: |         env: | ||||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} |           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} |           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_CAST_SITE_ID }} | ||||||
|   | |||||||
							
								
								
									
										22
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.github/workflows/ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -24,9 +24,9 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -37,7 +37,7 @@ jobs: | |||||||
|       - name: Build resources |       - name: Build resources | ||||||
|         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages |         run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages | ||||||
|       - name: Setup lint cache |       - name: Setup lint cache | ||||||
|         uses: actions/cache@v4.2.3 |         uses: actions/cache@0057852bfaa89a56745cba8c7296529d2fc39830 # v4.3.0 | ||||||
|         with: |         with: | ||||||
|           path: | |           path: | | ||||||
|             node_modules/.cache/prettier |             node_modules/.cache/prettier | ||||||
| @@ -58,9 +58,9 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -76,9 +76,9 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -89,7 +89,7 @@ jobs: | |||||||
|         env: |         env: | ||||||
|           IS_TEST: "true" |           IS_TEST: "true" | ||||||
|       - name: Upload bundle stats |       - name: Upload bundle stats | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: frontend-bundle-stats |           name: frontend-bundle-stats | ||||||
|           path: build/stats/*.json |           path: build/stats/*.json | ||||||
| @@ -100,9 +100,9 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -113,7 +113,7 @@ jobs: | |||||||
|         env: |         env: | ||||||
|           IS_TEST: "true" |           IS_TEST: "true" | ||||||
|       - name: Upload bundle stats |       - name: Upload bundle stats | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: supervisor-bundle-stats |           name: supervisor-bundle-stats | ||||||
|           path: build/stats/*.json |           path: build/stats/*.json | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @@ -23,7 +23,7 @@ jobs: | |||||||
|  |  | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout repository |       - name: Checkout repository | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           # We must fetch at least the immediate parents so that if this is |           # We must fetch at least the immediate parents so that if this is | ||||||
|           # a pull request then we can checkout the head. |           # a pull request then we can checkout the head. | ||||||
| @@ -36,14 +36,14 @@ jobs: | |||||||
|  |  | ||||||
|       # Initializes the CodeQL tools for scanning. |       # Initializes the CodeQL tools for scanning. | ||||||
|       - name: Initialize CodeQL |       - name: Initialize CodeQL | ||||||
|         uses: github/codeql-action/init@v3 |         uses: github/codeql-action/init@64d10c13136e1c5bce3e5fbde8d4906eeaafc885 # v3.30.6 | ||||||
|         with: |         with: | ||||||
|           languages: ${{ matrix.language }} |           languages: ${{ matrix.language }} | ||||||
|  |  | ||||||
|       # Autobuild attempts to build any compiled languages  (C/C++, C#, or Java). |       # Autobuild attempts to build any compiled languages  (C/C++, C#, or Java). | ||||||
|       # If this step fails, then you should remove it and run the build manually (see below) |       # If this step fails, then you should remove it and run the build manually (see below) | ||||||
|       - name: Autobuild |       - name: Autobuild | ||||||
|         uses: github/codeql-action/autobuild@v3 |         uses: github/codeql-action/autobuild@64d10c13136e1c5bce3e5fbde8d4906eeaafc885 # v3.30.6 | ||||||
|  |  | ||||||
|       # ℹ️ Command-line programs to run using the OS shell. |       # ℹ️ Command-line programs to run using the OS shell. | ||||||
|       # 📚 https://git.io/JvXDl |       # 📚 https://git.io/JvXDl | ||||||
| @@ -57,4 +57,4 @@ jobs: | |||||||
|       #   make release |       #   make release | ||||||
|  |  | ||||||
|       - name: Perform CodeQL Analysis |       - name: Perform CodeQL Analysis | ||||||
|         uses: github/codeql-action/analyze@v3 |         uses: github/codeql-action/analyze@64d10c13136e1c5bce3e5fbde8d4906eeaafc885 # v3.30.6 | ||||||
|   | |||||||
							
								
								
									
										12
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/demo_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -22,12 +22,12 @@ jobs: | |||||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} |       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           ref: dev |           ref: dev | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -43,7 +43,7 @@ jobs: | |||||||
|       - name: Deploy to Netlify |       - name: Deploy to Netlify | ||||||
|         id: deploy |         id: deploy | ||||||
|         run: | |         run: | | ||||||
|           npx -y netlify-cli deploy --dir=demo/dist --prod |           npx -y netlify-cli@23.7.3 deploy --dir=demo/dist --prod | ||||||
|         env: |         env: | ||||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} |           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_DEV_SITE_ID }} |           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_DEV_SITE_ID }} | ||||||
| @@ -57,12 +57,12 @@ jobs: | |||||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} |       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           ref: master |           ref: master | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -78,7 +78,7 @@ jobs: | |||||||
|       - name: Deploy to Netlify |       - name: Deploy to Netlify | ||||||
|         id: deploy |         id: deploy | ||||||
|         run: | |         run: | | ||||||
|           npx -y netlify-cli deploy --dir=demo/dist --prod |           npx -y netlify-cli@23.7.3 deploy --dir=demo/dist --prod | ||||||
|         env: |         env: | ||||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} |           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_SITE_ID }} |           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_DEMO_SITE_ID }} | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/design_deployment.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -16,10 +16,10 @@ jobs: | |||||||
|       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} |       url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }} | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -35,7 +35,7 @@ jobs: | |||||||
|       - name: Deploy to Netlify |       - name: Deploy to Netlify | ||||||
|         id: deploy |         id: deploy | ||||||
|         run: | |         run: | | ||||||
|           npx -y netlify-cli deploy --dir=gallery/dist --prod |           npx -y netlify-cli@23.7.3 deploy --dir=gallery/dist --prod | ||||||
|         env: |         env: | ||||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} |           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||||
|           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }} |           NETLIFY_SITE_ID: ${{ secrets.NETLIFY_GALLERY_SITE_ID }} | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/design_preview.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -21,10 +21,10 @@ jobs: | |||||||
|     if: github.repository == 'home-assistant/frontend' && contains(github.event.pull_request.labels.*.name, 'needs design preview') |     if: github.repository == 'home-assistant/frontend' && contains(github.event.pull_request.labels.*.name, 'needs design preview') | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out files from GitHub |       - name: Check out files from GitHub | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -40,7 +40,7 @@ jobs: | |||||||
|       - name: Deploy preview to Netlify |       - name: Deploy preview to Netlify | ||||||
|         id: deploy |         id: deploy | ||||||
|         run: | |         run: | | ||||||
|           npx -y netlify-cli deploy --dir=gallery/dist --alias "deploy-preview-${{ github.event.number }}" \ |           npx -y netlify-cli@23.7.3 deploy --dir=gallery/dist --alias "deploy-preview-${{ github.event.number }}" \ | ||||||
|             --json > deploy_output.json |             --json > deploy_output.json | ||||||
|         env: |         env: | ||||||
|           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} |           NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/labeler.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/labeler.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -10,6 +10,6 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Apply labels |       - name: Apply labels | ||||||
|         uses: actions/labeler@v5.0.0 |         uses: actions/labeler@634933edcd8ababfe52f92936142cc22ac488b1b # v6.0.1 | ||||||
|         with: |         with: | ||||||
|           sync-labels: true |           sync-labels: true | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,7 +9,7 @@ jobs: | |||||||
|   lock: |   lock: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: dessant/lock-threads@v5.0.1 |       - uses: dessant/lock-threads@1bf7ec25051fe7c00bdd17e6a7cf3d7bfb7dc771 # v5.0.1 | ||||||
|         with: |         with: | ||||||
|           github-token: ${{ github.token }} |           github-token: ${{ github.token }} | ||||||
|           process-only: "issues, prs" |           process-only: "issues, prs" | ||||||
|   | |||||||
							
								
								
									
										10
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/nightly.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -20,15 +20,15 @@ jobs: | |||||||
|       contents: write |       contents: write | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} |       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.PYTHON_VERSION }} |           python-version: ${{ env.PYTHON_VERSION }} | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -57,14 +57,14 @@ jobs: | |||||||
|         run: tar -czvf translations.tar.gz translations |         run: tar -czvf translations.tar.gz translations | ||||||
|  |  | ||||||
|       - name: Upload build artifacts |       - name: Upload build artifacts | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: wheels |           name: wheels | ||||||
|           path: dist/home_assistant_frontend*.whl |           path: dist/home_assistant_frontend*.whl | ||||||
|           if-no-files-found: error |           if-no-files-found: error | ||||||
|  |  | ||||||
|       - name: Upload translations |       - name: Upload translations | ||||||
|         uses: actions/upload-artifact@v4.6.2 |         uses: actions/upload-artifact@ea165f8d65b6e75b540449e92b4886f43607fa02 # v4.6.2 | ||||||
|         with: |         with: | ||||||
|           name: translations |           name: translations | ||||||
|           path: translations.tar.gz |           path: translations.tar.gz | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/relative-ci.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/relative-ci.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -17,7 +17,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Send bundle stats and build information to RelativeCI |       - name: Send bundle stats and build information to RelativeCI | ||||||
|         uses: relative-ci/agent-action@v3.0.0 |         uses: relative-ci/agent-action@1707825cbfcc7452b2913d273414705415ae64d4 # v3.0.1 | ||||||
|         with: |         with: | ||||||
|           key: ${{ secrets[format('RELATIVE_CI_KEY_{0}_{1}', matrix.bundle, matrix.build)] }} |           key: ${{ secrets[format('RELATIVE_CI_KEY_{0}_{1}', matrix.bundle, matrix.build)] }} | ||||||
|           token: ${{ github.token }} |           token: ${{ github.token }} | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/release-drafter.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/release-drafter.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -18,6 +18,6 @@ jobs: | |||||||
|       pull-requests: read |       pull-requests: read | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: release-drafter/release-drafter@v6.1.0 |       - uses: release-drafter/release-drafter@b1476f6e6eb133afa41ed8589daba6dc69b4d3f5 # v6.1.0 | ||||||
|         env: |         env: | ||||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} |           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||||
|   | |||||||
							
								
								
									
										23
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										23
									
								
								.github/workflows/release.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -23,10 +23,10 @@ jobs: | |||||||
|       contents: write # Required to upload release assets |       contents: write # Required to upload release assets | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Set up Python ${{ env.PYTHON_VERSION }} |       - name: Set up Python ${{ env.PYTHON_VERSION }} | ||||||
|         uses: actions/setup-python@v5 |         uses: actions/setup-python@e797f83bcb11b83ae66e0230d6156d7c80228e7c # v6.0.0 | ||||||
|         with: |         with: | ||||||
|           python-version: ${{ env.PYTHON_VERSION }} |           python-version: ${{ env.PYTHON_VERSION }} | ||||||
|  |  | ||||||
| @@ -34,7 +34,7 @@ jobs: | |||||||
|         uses: home-assistant/actions/helpers/verify-version@master |         uses: home-assistant/actions/helpers/verify-version@master | ||||||
|  |  | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -55,7 +55,7 @@ jobs: | |||||||
|           script/release |           script/release | ||||||
|  |  | ||||||
|       - name: Upload release assets |       - name: Upload release assets | ||||||
|         uses: softprops/action-gh-release@v2.3.2 |         uses: softprops/action-gh-release@62c96d0c4e8a889135c1f3a25910db8dbe0e85f7 # v2.3.4 | ||||||
|         with: |         with: | ||||||
|           files: | |           files: | | ||||||
|             dist/*.whl |             dist/*.whl | ||||||
| @@ -73,8 +73,9 @@ jobs: | |||||||
|           version=$(echo "${{ github.ref }}" | awk -F"/" '{print $NF}' ) |           version=$(echo "${{ github.ref }}" | awk -F"/" '{print $NF}' ) | ||||||
|           echo "home-assistant-frontend==$version" > ./requirements.txt |           echo "home-assistant-frontend==$version" > ./requirements.txt | ||||||
|  |  | ||||||
|  |       # home-assistant/wheels doesn't support SHA pinning | ||||||
|       - name: Build wheels |       - name: Build wheels | ||||||
|         uses: home-assistant/wheels@2025.03.0 |         uses: home-assistant/wheels@2025.09.1 | ||||||
|         with: |         with: | ||||||
|           abi: cp313 |           abi: cp313 | ||||||
|           tag: musllinux_1_2 |           tag: musllinux_1_2 | ||||||
| @@ -90,9 +91,9 @@ jobs: | |||||||
|       contents: write # Required to upload release assets |       contents: write # Required to upload release assets | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -107,7 +108,7 @@ jobs: | |||||||
|       - name: Tar folder |       - name: Tar folder | ||||||
|         run: tar -czf landing-page/home_assistant_frontend_landingpage-${{ github.event.release.tag_name }}.tar.gz -C landing-page/dist . |         run: tar -czf landing-page/home_assistant_frontend_landingpage-${{ github.event.release.tag_name }}.tar.gz -C landing-page/dist . | ||||||
|       - name: Upload release asset |       - name: Upload release asset | ||||||
|         uses: softprops/action-gh-release@v2.3.2 |         uses: softprops/action-gh-release@62c96d0c4e8a889135c1f3a25910db8dbe0e85f7 # v2.3.4 | ||||||
|         with: |         with: | ||||||
|           files: landing-page/home_assistant_frontend_landingpage-${{ github.event.release.tag_name }}.tar.gz |           files: landing-page/home_assistant_frontend_landingpage-${{ github.event.release.tag_name }}.tar.gz | ||||||
|  |  | ||||||
| @@ -119,9 +120,9 @@ jobs: | |||||||
|       contents: write # Required to upload release assets |       contents: write # Required to upload release assets | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|       - name: Setup Node |       - name: Setup Node | ||||||
|         uses: actions/setup-node@v4.4.0 |         uses: actions/setup-node@a0853c24544627f65ddf259abe73b1d18a591444 # v5.0.0 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ".nvmrc" |           node-version-file: ".nvmrc" | ||||||
|           cache: yarn |           cache: yarn | ||||||
| @@ -136,6 +137,6 @@ jobs: | |||||||
|       - name: Tar folder |       - name: Tar folder | ||||||
|         run: tar -czf hassio/home_assistant_frontend_supervisor-${{ github.event.release.tag_name }}.tar.gz -C hassio/build . |         run: tar -czf hassio/home_assistant_frontend_supervisor-${{ github.event.release.tag_name }}.tar.gz -C hassio/build . | ||||||
|       - name: Upload release asset |       - name: Upload release asset | ||||||
|         uses: softprops/action-gh-release@v2.3.2 |         uses: softprops/action-gh-release@62c96d0c4e8a889135c1f3a25910db8dbe0e85f7 # v2.3.4 | ||||||
|         with: |         with: | ||||||
|           files: hassio/home_assistant_frontend_supervisor-${{ github.event.release.tag_name }}.tar.gz |           files: hassio/home_assistant_frontend_supervisor-${{ github.event.release.tag_name }}.tar.gz | ||||||
|   | |||||||
							
								
								
									
										58
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | |||||||
|  | name: Restrict task creation | ||||||
|  |  | ||||||
|  | # yamllint disable-line rule:truthy | ||||||
|  | on: | ||||||
|  |   issues: | ||||||
|  |     types: [opened] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   check-authorization: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     # Only run if this is a Task issue type (from the issue form) | ||||||
|  |     if: github.event.issue.type.name == 'Task' | ||||||
|  |     steps: | ||||||
|  |       - name: Check if user is authorized | ||||||
|  |         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8 | ||||||
|  |         with: | ||||||
|  |           script: | | ||||||
|  |             const issueAuthor = context.payload.issue.user.login; | ||||||
|  |  | ||||||
|  |             // Check if user is an organization member | ||||||
|  |             try { | ||||||
|  |               await github.rest.orgs.checkMembershipForUser({ | ||||||
|  |                 org: 'home-assistant', | ||||||
|  |                 username: issueAuthor | ||||||
|  |               }); | ||||||
|  |               console.log(`✅ ${issueAuthor} is an organization member`); | ||||||
|  |               return; // Authorized | ||||||
|  |             } catch (error) { | ||||||
|  |               console.log(`❌ ${issueAuthor} is not authorized to create Task issues`); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             // Close the issue with a comment | ||||||
|  |             await github.rest.issues.createComment({ | ||||||
|  |               owner: context.repo.owner, | ||||||
|  |               repo: context.repo.repo, | ||||||
|  |               issue_number: context.issue.number, | ||||||
|  |               body: `Hi @${issueAuthor}, thank you for your contribution!\n\n` + | ||||||
|  |                     `Task issues are restricted to Open Home Foundation staff and authorized contributors.\n\n` + | ||||||
|  |                     `If you would like to:\n` + | ||||||
|  |                     `- Report a bug: Please use the [bug report form](https://github.com/home-assistant/frontend/issues/new?template=bug_report.yml)\n` + | ||||||
|  |                     `- Request a feature: Please submit to [Feature Requests](https://github.com/orgs/home-assistant/discussions)\n\n` + | ||||||
|  |                     `If you believe you should have access to create Task issues, please contact the maintainers.` | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             await github.rest.issues.update({ | ||||||
|  |               owner: context.repo.owner, | ||||||
|  |               repo: context.repo.repo, | ||||||
|  |               issue_number: context.issue.number, | ||||||
|  |               state: 'closed' | ||||||
|  |             }); | ||||||
|  |  | ||||||
|  |             // Add a label to indicate this was auto-closed | ||||||
|  |             await github.rest.issues.addLabels({ | ||||||
|  |               owner: context.repo.owner, | ||||||
|  |               repo: context.repo.repo, | ||||||
|  |               issue_number: context.issue.number, | ||||||
|  |               labels: ['auto-closed'] | ||||||
|  |             }); | ||||||
							
								
								
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,7 +10,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: 90 days stale policy |       - name: 90 days stale policy | ||||||
|         uses: actions/stale@v9.1.0 |         uses: actions/stale@5f858e3efba33a5ca4407a664cc011ad407f2008 # v10.1.0 | ||||||
|         with: |         with: | ||||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} |           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||||
|           days-before-stale: 90 |           days-before-stale: 90 | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,7 @@ | |||||||
| name: Translations | name: Translations | ||||||
|  |  | ||||||
| on: | on: | ||||||
|  |   workflow_dispatch: | ||||||
|   push: |   push: | ||||||
|     branches: |     branches: | ||||||
|       - dev |       - dev | ||||||
| @@ -13,7 +14,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout the repository |       - name: Checkout the repository | ||||||
|         uses: actions/checkout@v4.2.2 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Upload Translations |       - name: Upload Translations | ||||||
|         run: | |         run: | | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -53,3 +53,7 @@ src/cast/dev_const.ts | |||||||
|  |  | ||||||
| # test coverage | # test coverage | ||||||
| test/coverage/ | test/coverage/ | ||||||
|  |  | ||||||
|  | # AI tooling | ||||||
|  | .claude | ||||||
|  |  | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @@ -6,4 +6,4 @@ enableGlobalCache: false | |||||||
|  |  | ||||||
| nodeLinker: node-modules | nodeLinker: node-modules | ||||||
|  |  | ||||||
| yarnPath: .yarn/releases/yarn-4.9.2.cjs | yarnPath: .yarn/releases/yarn-4.10.3.cjs | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								CODEOWNERS
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								CODEOWNERS
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | # People marked here will be automatically requested for a review | ||||||
|  | # when the code that they own is touched. | ||||||
|  | # https://github.com/blog/2392-introducing-code-owners | ||||||
|  | # https://docs.github.com/en/repositories/managing-your-repositorys-settings-and-features/customizing-your-repository/about-code-owners | ||||||
|  |  | ||||||
|  | # Part of the frontend that mobile developper should review | ||||||
|  | src/external_app/ @bgoncal @TimoPtr | ||||||
|  | test/external_app/ @bgoncal @TimoPtr | ||||||
| @@ -183,7 +183,6 @@ module.exports.babelOptions = ({ | |||||||
|       include: /\/node_modules\//, |       include: /\/node_modules\//, | ||||||
|       exclude: [ |       exclude: [ | ||||||
|         "element-internals-polyfill", |         "element-internals-polyfill", | ||||||
|         "@shoelace-style", |  | ||||||
|         "@?lit(?:-labs|-element|-html)?", |         "@?lit(?:-labs|-element|-html)?", | ||||||
|       ].map((p) => new RegExp(`/node_modules/${p}/`)), |       ].map((p) => new RegExp(`/node_modules/${p}/`)), | ||||||
|     }, |     }, | ||||||
|   | |||||||
| @@ -14,5 +14,5 @@ | |||||||
|   "name": "Home Assistant Cast", |   "name": "Home Assistant Cast", | ||||||
|   "short_name": "HA Cast", |   "short_name": "HA Cast", | ||||||
|   "start_url": "/?homescreen=1", |   "start_url": "/?homescreen=1", | ||||||
|   "theme_color": "#03A9F4" |   "theme_color": "#009ac7" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,5 +1,3 @@ | |||||||
| import "@material/mwc-button/mwc-button"; |  | ||||||
|  |  | ||||||
| import type { ActionDetail } from "@material/mwc-list/mwc-list"; | import type { ActionDetail } from "@material/mwc-list/mwc-list"; | ||||||
| import { mdiCast, mdiCastConnected, mdiViewDashboard } from "@mdi/js"; | import { mdiCast, mdiCastConnected, mdiViewDashboard } from "@mdi/js"; | ||||||
| import type { Auth, Connection } from "home-assistant-js-websocket"; | import type { Auth, Connection } from "home-assistant-js-websocket"; | ||||||
| @@ -20,6 +18,7 @@ import { atLeastVersion } from "../../../../src/common/config/version"; | |||||||
| import { toggleAttribute } from "../../../../src/common/dom/toggle_attribute"; | import { toggleAttribute } from "../../../../src/common/dom/toggle_attribute"; | ||||||
| import "../../../../src/components/ha-icon"; | import "../../../../src/components/ha-icon"; | ||||||
| import "../../../../src/components/ha-list"; | import "../../../../src/components/ha-list"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-list-item"; | import "../../../../src/components/ha-list-item"; | ||||||
| import "../../../../src/components/ha-svg-icon"; | import "../../../../src/components/ha-svg-icon"; | ||||||
| import { | import { | ||||||
| @@ -63,12 +62,20 @@ class HcCast extends LitElement { | |||||||
|               <p class="question action-item"> |               <p class="question action-item"> | ||||||
|                 Stay logged in? |                 Stay logged in? | ||||||
|                 <span> |                 <span> | ||||||
|                   <mwc-button @click=${this._handleSaveTokens}> |                   <ha-button | ||||||
|  |                     appearance="plain" | ||||||
|  |                     size="small" | ||||||
|  |                     @click=${this._handleSaveTokens} | ||||||
|  |                   > | ||||||
|                     YES |                     YES | ||||||
|                   </mwc-button> |                   </ha-button> | ||||||
|                   <mwc-button @click=${this._handleSkipSaveTokens}> |                   <ha-button | ||||||
|  |                     appearance="plain" | ||||||
|  |                     size="small" | ||||||
|  |                     @click=${this._handleSkipSaveTokens} | ||||||
|  |                   > | ||||||
|                     NO |                     NO | ||||||
|                   </mwc-button> |                   </ha-button> | ||||||
|                 </span> |                 </span> | ||||||
|               </p> |               </p> | ||||||
|             ` |             ` | ||||||
| @@ -78,10 +85,10 @@ class HcCast extends LitElement { | |||||||
|           : !this.castManager.status |           : !this.castManager.status | ||||||
|             ? html` |             ? html` | ||||||
|                 <p class="center-item"> |                 <p class="center-item"> | ||||||
|                   <mwc-button raised @click=${this._handleLaunch}> |                   <ha-button @click=${this._handleLaunch}> | ||||||
|                     <ha-svg-icon .path=${mdiCast}></ha-svg-icon> |                     <ha-svg-icon slot="start" .path=${mdiCast}></ha-svg-icon> | ||||||
|                     Start Casting |                     Start Casting | ||||||
|                   </mwc-button> |                   </ha-button> | ||||||
|                 </p> |                 </p> | ||||||
|               ` |               ` | ||||||
|             : html` |             : html` | ||||||
| @@ -121,14 +128,22 @@ class HcCast extends LitElement { | |||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           ${this.castManager.status |           ${this.castManager.status | ||||||
|             ? html` |             ? html` | ||||||
|                 <mwc-button @click=${this._handleLaunch}> |                 <ha-button appearance="plain" @click=${this._handleLaunch}> | ||||||
|                   <ha-svg-icon .path=${mdiCastConnected}></ha-svg-icon> |                   <ha-svg-icon | ||||||
|  |                     slot="start" | ||||||
|  |                     .path=${mdiCastConnected} | ||||||
|  |                   ></ha-svg-icon> | ||||||
|                   Manage |                   Manage | ||||||
|                 </mwc-button> |                 </ha-button> | ||||||
|               ` |               ` | ||||||
|             : ""} |             : ""} | ||||||
|           <div class="spacer"></div> |           <div class="spacer"></div> | ||||||
|           <mwc-button @click=${this._handleLogout}>Log out</mwc-button> |           <ha-button | ||||||
|  |             variant="danger" | ||||||
|  |             appearance="plain" | ||||||
|  |             @click=${this._handleLogout} | ||||||
|  |             >Log out</ha-button | ||||||
|  |           > | ||||||
|         </div> |         </div> | ||||||
|       </hc-layout> |       </hc-layout> | ||||||
|     `; |     `; | ||||||
| @@ -227,7 +242,7 @@ class HcCast extends LitElement { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     .question:before { |     .question:before { | ||||||
|       border-radius: 4px; |       border-radius: var(--ha-border-radius-sm); | ||||||
|       position: absolute; |       position: absolute; | ||||||
|       top: 0; |       top: 0; | ||||||
|       right: 0; |       right: 0; | ||||||
| @@ -245,13 +260,6 @@ class HcCast extends LitElement { | |||||||
|       color: var(--secondary-text-color); |       color: var(--secondary-text-color); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     mwc-button ha-svg-icon { |  | ||||||
|       margin-right: 8px; |  | ||||||
|       margin-inline-end: 8px; |  | ||||||
|       margin-inline-start: initial; |  | ||||||
|       height: 18px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     ha-list-item ha-icon, |     ha-list-item ha-icon, | ||||||
|     ha-list-item ha-svg-icon { |     ha-list-item ha-svg-icon { | ||||||
|       padding: 12px; |       padding: 12px; | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
| import { mdiCastConnected, mdiCast } from "@mdi/js"; | import { mdiCastConnected, mdiCast } from "@mdi/js"; | ||||||
| import type { | import type { | ||||||
|   Auth, |   Auth, | ||||||
| @@ -28,6 +27,7 @@ import "../../../../src/layouts/hass-loading-screen"; | |||||||
| import { registerServiceWorker } from "../../../../src/util/register-service-worker"; | import { registerServiceWorker } from "../../../../src/util/register-service-worker"; | ||||||
| import "./hc-layout"; | import "./hc-layout"; | ||||||
| import "../../../../src/components/ha-textfield"; | import "../../../../src/components/ha-textfield"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
|  |  | ||||||
| const seeFAQ = (qid) => html` | const seeFAQ = (qid) => html` | ||||||
|   See <a href="./faq.html${qid ? `#${qid}` : ""}">the FAQ</a> for more |   See <a href="./faq.html${qid ? `#${qid}` : ""}">the FAQ</a> for more | ||||||
| @@ -83,11 +83,14 @@ export class HcConnect extends LitElement { | |||||||
|             Unable to connect to ${tokens!.hassUrl}. |             Unable to connect to ${tokens!.hassUrl}. | ||||||
|           </div> |           </div> | ||||||
|           <div class="card-actions"> |           <div class="card-actions"> | ||||||
|             <a href="/"> |             <ha-button appearance="plain" href="/">Retry</ha-button> | ||||||
|               <mwc-button> Retry </mwc-button> |  | ||||||
|             </a> |  | ||||||
|             <div class="spacer"></div> |             <div class="spacer"></div> | ||||||
|             <mwc-button @click=${this._handleLogout}>Log out</mwc-button> |             <ha-button | ||||||
|  |               appearance="plain" | ||||||
|  |               variant="danger" | ||||||
|  |               @click=${this._handleLogout} | ||||||
|  |               >Log out</ha-button | ||||||
|  |             > | ||||||
|           </div> |           </div> | ||||||
|         </hc-layout> |         </hc-layout> | ||||||
|       `; |       `; | ||||||
| @@ -128,16 +131,19 @@ export class HcConnect extends LitElement { | |||||||
|             ${this.error ? html` <p class="error">${this.error}</p> ` : ""} |             ${this.error ? html` <p class="error">${this.error}</p> ` : ""} | ||||||
|           </div> |           </div> | ||||||
|           <div class="card-actions"> |           <div class="card-actions"> | ||||||
|             <mwc-button @click=${this._handleDemo}> |             <ha-button appearance="plain" @click=${this._handleDemo}> | ||||||
|               Show Demo |               Show Demo | ||||||
|               <ha-svg-icon |               <ha-svg-icon | ||||||
|  |                 slot="end" | ||||||
|                 .path=${this.castManager.castState === "CONNECTED" |                 .path=${this.castManager.castState === "CONNECTED" | ||||||
|                   ? mdiCastConnected |                   ? mdiCastConnected | ||||||
|                   : mdiCast} |                   : mdiCast} | ||||||
|               ></ha-svg-icon> |               ></ha-svg-icon> | ||||||
|             </mwc-button> |             </ha-button> | ||||||
|             <div class="spacer"></div> |             <div class="spacer"></div> | ||||||
|             <mwc-button @click=${this._handleConnect}>Authorize</mwc-button> |             <ha-button appearance="plain" @click=${this._handleConnect} | ||||||
|  |               >Authorize</ha-button | ||||||
|  |             > | ||||||
|           </div> |           </div> | ||||||
|         </hc-layout> |         </hc-layout> | ||||||
|       `; |       `; | ||||||
| @@ -309,10 +315,6 @@ export class HcConnect extends LitElement { | |||||||
|       color: darkred; |       color: darkred; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     mwc-button ha-svg-icon { |  | ||||||
|       margin-left: 8px; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     .spacer { |     .spacer { | ||||||
|       flex: 1; |       flex: 1; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -95,7 +95,8 @@ class HcLayout extends LitElement { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     .hero { |     .hero { | ||||||
|       border-radius: 4px 4px 0 0; |       border-radius: var(--ha-border-radius-sm) var(--ha-border-radius-sm) | ||||||
|  |         var(--ha-border-radius-square) var(--ha-border-radius-square); | ||||||
|     } |     } | ||||||
|     .subtitle { |     .subtitle { | ||||||
|       font-size: var(--ha-font-size-m); |       font-size: var(--ha-font-size-m); | ||||||
|   | |||||||
| @@ -75,7 +75,7 @@ export const castDemoEntities: () => Entity[] = () => | |||||||
|         longitude: 4.8903147, |         longitude: 4.8903147, | ||||||
|         radius: 100, |         radius: 100, | ||||||
|         friendly_name: "Home", |         friendly_name: "Home", | ||||||
|         icon: "hass:home", |         icon: "mdi:home", | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     "input_number.harmonyvolume": { |     "input_number.harmonyvolume": { | ||||||
| @@ -88,7 +88,7 @@ export const castDemoEntities: () => Entity[] = () => | |||||||
|         step: 1, |         step: 1, | ||||||
|         mode: "slider", |         mode: "slider", | ||||||
|         friendly_name: "Volume", |         friendly_name: "Volume", | ||||||
|         icon: "hass:volume-high", |         icon: "mdi:volume-high", | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     "climate.upstairs": { |     "climate.upstairs": { | ||||||
|   | |||||||
| @@ -56,7 +56,7 @@ export const castDemoLovelace: () => LovelaceConfig = () => { | |||||||
|                 type: "weblink", |                 type: "weblink", | ||||||
|                 url: "/lovelace/climate", |                 url: "/lovelace/climate", | ||||||
|                 name: "Climate controls", |                 name: "Climate controls", | ||||||
|                 icon: "hass:arrow-right", |                 icon: "mdi:arrow-right", | ||||||
|               }, |               }, | ||||||
|             ], |             ], | ||||||
|           }, |           }, | ||||||
| @@ -76,7 +76,7 @@ export const castDemoLovelace: () => LovelaceConfig = () => { | |||||||
|                 type: "weblink", |                 type: "weblink", | ||||||
|                 url: "/lovelace/overview", |                 url: "/lovelace/overview", | ||||||
|                 name: "Back", |                 name: "Back", | ||||||
|                 icon: "hass:arrow-left", |                 icon: "mdi:arrow-left", | ||||||
|               }, |               }, | ||||||
|             ], |             ], | ||||||
|           }, |           }, | ||||||
|   | |||||||
| @@ -75,5 +75,5 @@ | |||||||
|   "name": "Home Assistant Demo", |   "name": "Home Assistant Demo", | ||||||
|   "short_name": "HA Demo", |   "short_name": "HA Demo", | ||||||
|   "start_url": "/?homescreen=1", |   "start_url": "/?homescreen=1", | ||||||
|   "theme_color": "#03A9F4" |   "theme_color": "#009ac7" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -143,7 +143,7 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) => | |||||||
|       state: "on", |       state: "on", | ||||||
|       attributes: { |       attributes: { | ||||||
|         friendly_name: "Home Automation", |         friendly_name: "Home Automation", | ||||||
|         icon: "hass:home-automation", |         icon: "mdi:home-automation", | ||||||
|       }, |       }, | ||||||
|     }, |     }, | ||||||
|     "input_boolean.tvtime": { |     "input_boolean.tvtime": { | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ export const demoLovelaceArsaboo: DemoConfig["lovelace"] = (localize) => ({ | |||||||
|   title: "Home Assistant", |   title: "Home Assistant", | ||||||
|   views: [ |   views: [ | ||||||
|     { |     { | ||||||
|       icon: "hass:home-assistant", |       icon: "mdi:home-assistant", | ||||||
|       id: "home", |       id: "home", | ||||||
|       title: "Home", |       title: "Home", | ||||||
|       cards: [ |       cards: [ | ||||||
|   | |||||||
| @@ -1236,7 +1236,7 @@ export const demoLovelaceJimpower: DemoConfig["lovelace"] = () => ({ | |||||||
|         }, |         }, | ||||||
|       ], |       ], | ||||||
|       path: "security", |       path: "security", | ||||||
|       icon: "hass:shield-home", |       icon: "mdi:shield-home", | ||||||
|       name: "Security", |       name: "Security", | ||||||
|       background: |       background: | ||||||
|         'center / cover no-repeat url("/assets/jimpower/background-15.jpg") fixed', |         'center / cover no-repeat url("/assets/jimpower/background-15.jpg") fixed', | ||||||
|   | |||||||
| @@ -89,11 +89,14 @@ export class HADemoCard extends LitElement implements LovelaceCard { | |||||||
|           )} |           )} | ||||||
|         </div> |         </div> | ||||||
|         <div class="actions small-hidden"> |         <div class="actions small-hidden"> | ||||||
|           <a href="https://www.home-assistant.io" target="_blank"> |           <ha-button | ||||||
|             <ha-button> |             appearance="plain" | ||||||
|               ${this.hass.localize("ui.panel.page-demo.cards.demo.learn_more")} |             size="small" | ||||||
|             </ha-button> |             href="https://www.home-assistant.io" | ||||||
|           </a> |             target="_blank" | ||||||
|  |           > | ||||||
|  |             ${this.hass.localize("ui.panel.page-demo.cards.demo.learn_more")} | ||||||
|  |           </ha-button> | ||||||
|         </div> |         </div> | ||||||
|       </ha-card> |       </ha-card> | ||||||
|     `; |     `; | ||||||
|   | |||||||
| @@ -68,7 +68,7 @@ | |||||||
|       } |       } | ||||||
|       #ha-launch-screen .ha-launch-screen-spacer-top { |       #ha-launch-screen .ha-launch-screen-spacer-top { | ||||||
|         flex: 1; |         flex: 1; | ||||||
|         margin-top: calc( 2 * max(var(--safe-area-inset-bottom), 48px) + 46px ); |         margin-top: calc( 2 * max(var(--safe-area-inset-top, 0px), 48px) + 46px ); | ||||||
|         padding-top: 48px; |         padding-top: 48px; | ||||||
|       } |       } | ||||||
|       #ha-launch-screen .ha-launch-screen-spacer-bottom { |       #ha-launch-screen .ha-launch-screen-spacer-bottom { | ||||||
| @@ -76,7 +76,7 @@ | |||||||
|         padding-top: 48px; |         padding-top: 48px; | ||||||
|       } |       } | ||||||
|       .ohf-logo { |       .ohf-logo { | ||||||
|         margin: max(var(--safe-area-inset-bottom), 48px) 0; |         margin: max(var(--safe-area-inset-bottom, 0px), 48px) 0; | ||||||
|         display: flex; |         display: flex; | ||||||
|         flex-direction: column; |         flex-direction: column; | ||||||
|         align-items: center; |         align-items: center; | ||||||
|   | |||||||
| @@ -1,11 +1,11 @@ | |||||||
| import "@material/mwc-button/mwc-button"; |  | ||||||
| import type { Button } from "@material/mwc-button"; |  | ||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { html, LitElement, css, nothing } from "lit"; | import { html, LitElement, css, nothing } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property } from "lit/decorators"; | ||||||
| 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 "../../../src/components/ha-card"; | import "../../../src/components/ha-card"; | ||||||
|  | import "../../../src/components/ha-button"; | ||||||
|  | import type { HaButton } from "../../../src/components/ha-button"; | ||||||
|  |  | ||||||
| @customElement("demo-black-white-row") | @customElement("demo-black-white-row") | ||||||
| class DemoBlackWhiteRow extends LitElement { | class DemoBlackWhiteRow extends LitElement { | ||||||
| @@ -25,12 +25,9 @@ class DemoBlackWhiteRow extends LitElement { | |||||||
|               <slot name="light"></slot> |               <slot name="light"></slot> | ||||||
|             </div> |             </div> | ||||||
|             <div class="card-actions"> |             <div class="card-actions"> | ||||||
|               <mwc-button |               <ha-button .disabled=${this.disabled} @click=${this.handleSubmit}> | ||||||
|                 .disabled=${this.disabled} |  | ||||||
|                 @click=${this.handleSubmit} |  | ||||||
|               > |  | ||||||
|                 Submit |                 Submit | ||||||
|               </mwc-button> |               </ha-button> | ||||||
|             </div> |             </div> | ||||||
|           </ha-card> |           </ha-card> | ||||||
|         </div> |         </div> | ||||||
| @@ -40,12 +37,9 @@ class DemoBlackWhiteRow extends LitElement { | |||||||
|               <slot name="dark"></slot> |               <slot name="dark"></slot> | ||||||
|             </div> |             </div> | ||||||
|             <div class="card-actions"> |             <div class="card-actions"> | ||||||
|               <mwc-button |               <ha-button .disabled=${this.disabled} @click=${this.handleSubmit}> | ||||||
|                 .disabled=${this.disabled} |  | ||||||
|                 @click=${this.handleSubmit} |  | ||||||
|               > |  | ||||||
|                 Submit |                 Submit | ||||||
|               </mwc-button> |               </ha-button> | ||||||
|             </div> |             </div> | ||||||
|           </ha-card> |           </ha-card> | ||||||
|           ${this.value |           ${this.value | ||||||
| @@ -74,7 +68,7 @@ class DemoBlackWhiteRow extends LitElement { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   handleSubmit(ev) { |   handleSubmit(ev) { | ||||||
|     const content = (ev.target as Button).closest(".content")!; |     const content = (ev.target as HaButton).closest(".content")!; | ||||||
|     fireEvent(this, "submitted" as any, { |     fireEvent(this, "submitted" as any, { | ||||||
|       slot: content.classList.contains("light") ? "light" : "dark", |       slot: content.classList.contains("light") ? "light" : "dark", | ||||||
|     }); |     }); | ||||||
|   | |||||||
| @@ -1,10 +1,11 @@ | |||||||
| import { LitElement, css, html } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property } from "lit/decorators"; | ||||||
| import "../../../src/components/ha-card"; | import "../../../src/components/ha-card"; | ||||||
| import "../../../src/dialogs/more-info/more-info-content"; | import "../../../src/dialogs/more-info/more-info-content"; | ||||||
| import "../../../src/state-summary/state-card-content"; | import "../../../src/state-summary/state-card-content"; | ||||||
| import "../ha-demo-options"; | import "../ha-demo-options"; | ||||||
| import type { HomeAssistant } from "../../../src/types"; | import type { HomeAssistant } from "../../../src/types"; | ||||||
|  | import { computeShowNewMoreInfo } from "../../../src/dialogs/more-info/const"; | ||||||
|  |  | ||||||
| @customElement("demo-more-info") | @customElement("demo-more-info") | ||||||
| class DemoMoreInfo extends LitElement { | class DemoMoreInfo extends LitElement { | ||||||
| @@ -21,11 +22,13 @@ class DemoMoreInfo extends LitElement { | |||||||
|       <div class="root"> |       <div class="root"> | ||||||
|         <div id="card"> |         <div id="card"> | ||||||
|           <ha-card> |           <ha-card> | ||||||
|             <state-card-content |             ${!computeShowNewMoreInfo(state) | ||||||
|               .stateObj=${state} |               ? html`<state-card-content | ||||||
|               .hass=${this.hass} |                   .stateObj=${state} | ||||||
|               in-dialog |                   .hass=${this.hass} | ||||||
|             ></state-card-content> |                   in-dialog | ||||||
|  |                 ></state-card-content>` | ||||||
|  |               : nothing} | ||||||
|  |  | ||||||
|             <more-info-content |             <more-info-content | ||||||
|               .hass=${this.hass} |               .hass=${this.hass} | ||||||
|   | |||||||
| @@ -1106,7 +1106,7 @@ export default { | |||||||
|       friendly_name: "Philips Hue", |       friendly_name: "Philips Hue", | ||||||
|       entity_picture: null, |       entity_picture: null, | ||||||
|       description: |       description: | ||||||
|         "Press the button on the bridge to register Philips Hue with Home Assistant.\n\n", |         "Press the button on the bridge to register Philips Hue with Home Assistant.", | ||||||
|       submit_caption: "I have pressed the button", |       submit_caption: "I have pressed the button", | ||||||
|     }, |     }, | ||||||
|     last_changed: "2018-07-19T10:44:46.515160+00:00", |     last_changed: "2018-07-19T10:44:46.515160+00:00", | ||||||
|   | |||||||
| @@ -17,6 +17,10 @@ export const createMediaPlayerEntities = () => [ | |||||||
|       new Date().getTime() - 23000 |       new Date().getTime() - 23000 | ||||||
|     ).toISOString(), |     ).toISOString(), | ||||||
|     volume_level: 0.5, |     volume_level: 0.5, | ||||||
|  |     source_list: ["AirPlay", "Blu-Ray", "TV", "USB", "iPod (USB)"], | ||||||
|  |     source: "AirPlay", | ||||||
|  |     sound_mode_list: ["Movie", "Music", "Game", "Pure Audio"], | ||||||
|  |     sound_mode: "Music", | ||||||
|   }), |   }), | ||||||
|   getEntity("media_player", "music_playing", "playing", { |   getEntity("media_player", "music_playing", "playing", { | ||||||
|     friendly_name: "Playing The Music", |     friendly_name: "Playing The Music", | ||||||
| @@ -24,8 +28,8 @@ export const createMediaPlayerEntities = () => [ | |||||||
|     media_title: "I Wanna Be A Hippy (Flamman & Abraxas Radio Mix)", |     media_title: "I Wanna Be A Hippy (Flamman & Abraxas Radio Mix)", | ||||||
|     media_artist: "Technohead", |     media_artist: "Technohead", | ||||||
|     // Pause + Seek + Volume Set + Volume Mute + Previous Track + Next Track + Play Media + |     // Pause + Seek + Volume Set + Volume Mute + Previous Track + Next Track + Play Media + | ||||||
|     // Select Source + Stop + Clear + Play + Shuffle Set + Browse Media |     // Select Source + Stop + Clear + Play + Shuffle Set + Browse Media + Grouping | ||||||
|     supported_features: 195135, |     supported_features: 784959, | ||||||
|     entity_picture: "/images/album_cover.jpg", |     entity_picture: "/images/album_cover.jpg", | ||||||
|     media_duration: 300, |     media_duration: 300, | ||||||
|     media_position: 0, |     media_position: 0, | ||||||
| @@ -34,6 +38,9 @@ export const createMediaPlayerEntities = () => [ | |||||||
|       new Date().getTime() - 23000 |       new Date().getTime() - 23000 | ||||||
|     ).toISOString(), |     ).toISOString(), | ||||||
|     volume_level: 0.5, |     volume_level: 0.5, | ||||||
|  |     sound_mode_list: ["Movie", "Music", "Game", "Pure Audio"], | ||||||
|  |     sound_mode: "Music", | ||||||
|  |     group_members: ["media_player.playing", "media_player.stream_playing"], | ||||||
|   }), |   }), | ||||||
|   getEntity("media_player", "stream_playing", "playing", { |   getEntity("media_player", "stream_playing", "playing", { | ||||||
|     friendly_name: "Playing the Stream", |     friendly_name: "Playing the Stream", | ||||||
| @@ -149,15 +156,18 @@ export const createMediaPlayerEntities = () => [ | |||||||
|   }), |   }), | ||||||
|   getEntity("media_player", "receiver_on", "on", { |   getEntity("media_player", "receiver_on", "on", { | ||||||
|     source_list: ["AirPlay", "Blu-Ray", "TV", "USB", "iPod (USB)"], |     source_list: ["AirPlay", "Blu-Ray", "TV", "USB", "iPod (USB)"], | ||||||
|  |     sound_mode_list: ["Movie", "Music", "Game", "Pure Audio"], | ||||||
|     volume_level: 0.63, |     volume_level: 0.63, | ||||||
|     is_volume_muted: false, |     is_volume_muted: false, | ||||||
|     source: "TV", |     source: "TV", | ||||||
|  |     sound_mode: "Movie", | ||||||
|     friendly_name: "Receiver (selectable sources)", |     friendly_name: "Receiver (selectable sources)", | ||||||
|     // Volume Set + Volume Mute + On + Off + Select Source + Play + Sound Mode |     // Volume Set + Volume Mute + On + Off + Select Source + Play + Sound Mode | ||||||
|     supported_features: 84364, |     supported_features: 84364, | ||||||
|   }), |   }), | ||||||
|   getEntity("media_player", "receiver_off", "off", { |   getEntity("media_player", "receiver_off", "off", { | ||||||
|     source_list: ["AirPlay", "Blu-Ray", "TV", "USB", "iPod (USB)"], |     source_list: ["AirPlay", "Blu-Ray", "TV", "USB", "iPod (USB)"], | ||||||
|  |     sound_mode_list: ["Movie", "Music", "Game", "Pure Audio"], | ||||||
|     friendly_name: "Receiver (selectable sources)", |     friendly_name: "Receiver (selectable sources)", | ||||||
|     // Volume Set + Volume Mute + On + Off + Select Source + Play + Sound Mode |     // Volume Set + Volume Mute + On + Off + Select Source + Play + Sound Mode | ||||||
|     supported_features: 84364, |     supported_features: 84364, | ||||||
|   | |||||||
| @@ -208,7 +208,7 @@ class HaGallery extends LitElement { | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       .sidebar a[active]::before { |       .sidebar a[active]::before { | ||||||
|         border-radius: 12px; |         border-radius: var(--ha-border-radius-lg); | ||||||
|         position: absolute; |         position: absolute; | ||||||
|         top: 0; |         top: 0; | ||||||
|         right: 2px; |         right: 2px; | ||||||
| @@ -241,7 +241,7 @@ class HaGallery extends LitElement { | |||||||
|         text-align: center; |         text-align: center; | ||||||
|         margin: 16px; |         margin: 16px; | ||||||
|         padding: 16px; |         padding: 16px; | ||||||
|         border-radius: 12px; |         border-radius: var(--ha-border-radius-lg); | ||||||
|         background-color: var(--primary-background-color); |         background-color: var(--primary-background-color); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -18,7 +18,6 @@ import { HaDeviceAction } from "../../../../src/panels/config/automation/action/ | |||||||
| import { HaEventAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-event"; | import { HaEventAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-event"; | ||||||
| import { HaIfAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-if"; | import { HaIfAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-if"; | ||||||
| import { HaParallelAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-parallel"; | import { HaParallelAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-parallel"; | ||||||
| import { HaPlayMediaAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-play_media"; |  | ||||||
| import { HaRepeatAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-repeat"; | import { HaRepeatAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-repeat"; | ||||||
| import { HaSequenceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-sequence"; | import { HaSequenceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-sequence"; | ||||||
| import { HaServiceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-service"; | import { HaServiceAction } from "../../../../src/panels/config/automation/action/types/ha-automation-action-service"; | ||||||
| @@ -32,7 +31,6 @@ const SCHEMAS: { name: string; actions: Action[] }[] = [ | |||||||
|   { name: "Service", actions: [HaServiceAction.defaultConfig] }, |   { name: "Service", actions: [HaServiceAction.defaultConfig] }, | ||||||
|   { name: "Condition", actions: [HaConditionAction.defaultConfig] }, |   { name: "Condition", actions: [HaConditionAction.defaultConfig] }, | ||||||
|   { name: "Delay", actions: [HaDelayAction.defaultConfig] }, |   { name: "Delay", actions: [HaDelayAction.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] }, | ||||||
|   | |||||||
| @@ -147,13 +147,13 @@ The `title ` option should not be used without a description. | |||||||
|  |  | ||||||
| <ha-alert alert-type="success"> | <ha-alert alert-type="success"> | ||||||
|   This is a success alert — check it out! |   This is a success alert — check it out! | ||||||
|   <mwc-button slot="action" label="Undo"></mwc-button> |   <ha-button slot="action">Undo</ha-button> | ||||||
| </ha-alert> | </ha-alert> | ||||||
|  |  | ||||||
| ```html | ```html | ||||||
| <ha-alert alert-type="success"> | <ha-alert alert-type="success"> | ||||||
|   This is a success alert — check it out! |   This is a success alert — check it out! | ||||||
|   <mwc-button slot="action" label="Undo"></mwc-button> |   <ha-button slot="action">Undo</ha-button> | ||||||
| </ha-alert> | </ha-alert> | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| import "@material/mwc-button/mwc-button"; |  | ||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement } from "lit/decorators"; | import { customElement } from "lit/decorators"; | ||||||
| import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||||
| import "../../../../src/components/ha-alert"; | import "../../../../src/components/ha-alert"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-logo-svg"; | import "../../../../src/components/ha-logo-svg"; | ||||||
|  |  | ||||||
| const alerts: { | const alerts: { | ||||||
| @@ -78,13 +78,13 @@ const alerts: { | |||||||
|     title: "Error with action", |     title: "Error with action", | ||||||
|     description: "This is a test error alert with action", |     description: "This is a test error alert with action", | ||||||
|     type: "error", |     type: "error", | ||||||
|     actionSlot: html`<mwc-button slot="action" label="restart"></mwc-button>`, |     actionSlot: html`<ha-button size="small" slot="action">restart</ha-button>`, | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     title: "Unsaved data", |     title: "Unsaved data", | ||||||
|     description: "You have unsaved data", |     description: "You have unsaved data", | ||||||
|     type: "warning", |     type: "warning", | ||||||
|     actionSlot: html`<mwc-button slot="action" label="save"></mwc-button>`, |     actionSlot: html`<ha-button size="small" slot="action">save</ha-button>`, | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     title: "Slotted icon", |     title: "Slotted icon", | ||||||
| @@ -108,7 +108,7 @@ const alerts: { | |||||||
|     title: "Slotted action", |     title: "Slotted action", | ||||||
|     description: "Alert with slotted action", |     description: "Alert with slotted action", | ||||||
|     type: "info", |     type: "info", | ||||||
|     actionSlot: html`<mwc-button slot="action" label="action"></mwc-button>`, |     actionSlot: html`<ha-button slot="action">action</ha-button>`, | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
|     description: "Dismissable information (RTL)", |     description: "Dismissable information (RTL)", | ||||||
| @@ -120,7 +120,7 @@ const alerts: { | |||||||
|     title: "Error with action", |     title: "Error with action", | ||||||
|     description: "This is a test error alert with action (RTL)", |     description: "This is a test error alert with action (RTL)", | ||||||
|     type: "error", |     type: "error", | ||||||
|     actionSlot: html`<mwc-button slot="action" label="restart"></mwc-button>`, |     actionSlot: html`<ha-button slot="action">restart</ha-button>`, | ||||||
|     rtl: true, |     rtl: true, | ||||||
|   }, |   }, | ||||||
|   { |   { | ||||||
| @@ -211,7 +211,7 @@ export class DemoHaAlert extends LitElement { | |||||||
|       max-height: 24px; |       max-height: 24px; | ||||||
|       width: 24px; |       width: 24px; | ||||||
|     } |     } | ||||||
|     mwc-button { |     ha-button { | ||||||
|       --mdc-theme-primary: var(--primary-text-color); |       --mdc-theme-primary: var(--primary-text-color); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
|   | |||||||
| @@ -117,7 +117,7 @@ export class DemoHaBadge extends LitElement { | |||||||
|     } |     } | ||||||
|     .card-content { |     .card-content { | ||||||
|       display: flex; |       display: flex; | ||||||
|       gap: 24px; |       gap: var(--ha-space-6); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										67
									
								
								gallery/src/pages/components/ha-button.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								gallery/src/pages/components/ha-button.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,67 @@ | |||||||
|  | --- | ||||||
|  | title: Button | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   .wrapper { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 24px; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  | # Button `<ha-button>` | ||||||
|  |  | ||||||
|  | ## Implementation | ||||||
|  |  | ||||||
|  | ### Example Usage | ||||||
|  |  | ||||||
|  | <div class="wrapper"> | ||||||
|  |   <ha-button> | ||||||
|  |     simple button | ||||||
|  |   </ha-button> | ||||||
|  |   <ha-button appearance="plain"> | ||||||
|  |     plain button | ||||||
|  |   </ha-button> | ||||||
|  |   <ha-button appearance="filled"> | ||||||
|  |     filled button | ||||||
|  |   </ha-button> | ||||||
|  |  | ||||||
|  |   <ha-button size="small"> | ||||||
|  |     small | ||||||
|  |   </ha-button> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <ha-button> simple button </ha-button> | ||||||
|  |  | ||||||
|  | <ha-button size="small"> small </ha-button> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### API | ||||||
|  |  | ||||||
|  | This component is based on the webawesome button component. | ||||||
|  | Check the [webawesome documentation](https://webawesome.com/docs/components/button/) for more details. | ||||||
|  |  | ||||||
|  | **Slots** | ||||||
|  |  | ||||||
|  | - default slot: Label of the button | ||||||
|  |   ` - no default | ||||||
|  | - `start`: The prefix container (usually for icons). | ||||||
|  |   ` - no default | ||||||
|  | - `end`: The suffix container (usually for icons). | ||||||
|  |   ` - no default | ||||||
|  |  | ||||||
|  | **Properties/Attributes** | ||||||
|  |  | ||||||
|  | | Name       | Type                                           | Default  | Description                                                                       | | ||||||
|  | | ---------- | ---------------------------------------------- | -------- | --------------------------------------------------------------------------------- | | ||||||
|  | | appearance | "accent"/"filled"/"plain"                      | "accent" | Sets the button appearance.                                                       | | ||||||
|  | | variants   | "brand"/"danger"/"neutral"/"warning"/"success" | "brand"  | Sets the button color variant. "brand" is default.                                | | ||||||
|  | | size       | "small"/"medium"                               | "medium" | Sets the button size.                                                             | | ||||||
|  | | loading    | Boolean                                        | false    | Shows a loading indicator instead of the buttons label and disable buttons click. | | ||||||
|  | | disabled   | Boolean                                        | false    | Disables the button and prevents user interaction.                                | | ||||||
|  |  | ||||||
|  | **CSS Custom Properties** | ||||||
|  |  | ||||||
|  | - `--ha-button-height` - Height of the button. | ||||||
|  | - `--ha-button-border-radius` - Border radius of the button. Defaults to `var(--ha-border-radius-pill)`. | ||||||
							
								
								
									
										171
									
								
								gallery/src/pages/components/ha-button.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										171
									
								
								gallery/src/pages/components/ha-button.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,171 @@ | |||||||
|  | import { mdiHome } from "@mdi/js"; | ||||||
|  | import type { TemplateResult } from "lit"; | ||||||
|  | import { css, html, LitElement } from "lit"; | ||||||
|  | import { customElement } from "lit/decorators"; | ||||||
|  | import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||||
|  | import { titleCase } from "../../../../src/common/string/title-case"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-svg-icon"; | ||||||
|  | import { mdiHomeAssistant } from "../../../../src/resources/home-assistant-logo-svg"; | ||||||
|  |  | ||||||
|  | const appearances = ["accent", "filled", "plain"]; | ||||||
|  | const variants = ["brand", "danger", "neutral", "warning", "success"]; | ||||||
|  |  | ||||||
|  | @customElement("demo-components-ha-button") | ||||||
|  | export class DemoHaButton extends LitElement { | ||||||
|  |   protected render(): TemplateResult { | ||||||
|  |     return html` | ||||||
|  |       ${["light", "dark"].map( | ||||||
|  |         (mode) => html` | ||||||
|  |           <div class=${mode}> | ||||||
|  |             <ha-card header="ha-button in ${mode}"> | ||||||
|  |               <div class="card-content"> | ||||||
|  |                 ${variants.map( | ||||||
|  |                   (variant) => html` | ||||||
|  |                     <div> | ||||||
|  |                       ${appearances.map( | ||||||
|  |                         (appearance) => html` | ||||||
|  |                           <ha-button | ||||||
|  |                             .appearance=${appearance} | ||||||
|  |                             .variant=${variant} | ||||||
|  |                           > | ||||||
|  |                             <ha-svg-icon | ||||||
|  |                               .path=${mdiHomeAssistant} | ||||||
|  |                               slot="start" | ||||||
|  |                             ></ha-svg-icon> | ||||||
|  |                             ${titleCase(`${variant} ${appearance}`)} | ||||||
|  |                             <ha-svg-icon | ||||||
|  |                               .path=${mdiHome} | ||||||
|  |                               slot="end" | ||||||
|  |                             ></ha-svg-icon> | ||||||
|  |                           </ha-button> | ||||||
|  |                         ` | ||||||
|  |                       )} | ||||||
|  |                     </div> | ||||||
|  |                     <div> | ||||||
|  |                       ${appearances.map( | ||||||
|  |                         (appearance) => html` | ||||||
|  |                           <ha-button | ||||||
|  |                             .appearance=${appearance} | ||||||
|  |                             .variant=${variant} | ||||||
|  |                             size="small" | ||||||
|  |                           > | ||||||
|  |                             ${titleCase(`${variant} ${appearance}`)} | ||||||
|  |                           </ha-button> | ||||||
|  |                         ` | ||||||
|  |                       )} | ||||||
|  |                     </div> | ||||||
|  |                     <div> | ||||||
|  |                       ${appearances.map( | ||||||
|  |                         (appearance) => html` | ||||||
|  |                           <ha-button | ||||||
|  |                             .appearance=${appearance} | ||||||
|  |                             .variant=${variant} | ||||||
|  |                             loading | ||||||
|  |                           > | ||||||
|  |                             <ha-svg-icon | ||||||
|  |                               .path=${mdiHomeAssistant} | ||||||
|  |                               slot="start" | ||||||
|  |                             ></ha-svg-icon> | ||||||
|  |                             ${titleCase(`${variant} ${appearance}`)} | ||||||
|  |                             <ha-svg-icon | ||||||
|  |                               .path=${mdiHome} | ||||||
|  |                               slot="end" | ||||||
|  |                             ></ha-svg-icon> | ||||||
|  |                           </ha-button> | ||||||
|  |                         ` | ||||||
|  |                       )} | ||||||
|  |                     </div> | ||||||
|  |                   ` | ||||||
|  |                 )} | ||||||
|  |                 ${variants.map( | ||||||
|  |                   (variant) => html` | ||||||
|  |                     <div> | ||||||
|  |                       ${appearances.map( | ||||||
|  |                         (appearance) => html` | ||||||
|  |                           <ha-button | ||||||
|  |                             .variant=${variant} | ||||||
|  |                             .appearance=${appearance} | ||||||
|  |                             disabled | ||||||
|  |                           > | ||||||
|  |                             ${titleCase(`${appearance}`)} | ||||||
|  |                           </ha-button> | ||||||
|  |                         ` | ||||||
|  |                       )} | ||||||
|  |                     </div> | ||||||
|  |                     <div> | ||||||
|  |                       ${appearances.map( | ||||||
|  |                         (appearance) => html` | ||||||
|  |                           <ha-button | ||||||
|  |                             .variant=${variant} | ||||||
|  |                             .appearance=${appearance} | ||||||
|  |                             size="small" | ||||||
|  |                             disabled | ||||||
|  |                           > | ||||||
|  |                             ${titleCase(`${appearance}`)} | ||||||
|  |                           </ha-button> | ||||||
|  |                         ` | ||||||
|  |                       )} | ||||||
|  |                     </div> | ||||||
|  |                   ` | ||||||
|  |                 )} | ||||||
|  |               </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 styles = css` | ||||||
|  |     :host { | ||||||
|  |       display: flex; | ||||||
|  |       justify-content: center; | ||||||
|  |     } | ||||||
|  |     .dark, | ||||||
|  |     .light { | ||||||
|  |       display: block; | ||||||
|  |       background-color: var(--primary-background-color); | ||||||
|  |       padding: 0 50px; | ||||||
|  |     } | ||||||
|  |     .button { | ||||||
|  |       padding: unset; | ||||||
|  |     } | ||||||
|  |     ha-card { | ||||||
|  |       margin: 24px auto; | ||||||
|  |     } | ||||||
|  |     .card-content { | ||||||
|  |       display: flex; | ||||||
|  |       flex-direction: column; | ||||||
|  |       gap: var(--ha-space-6); | ||||||
|  |     } | ||||||
|  |     .card-content div { | ||||||
|  |       display: flex; | ||||||
|  |       gap: var(--ha-space-2); | ||||||
|  |     } | ||||||
|  |   `; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-components-ha-button": DemoHaButton; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -9,10 +9,10 @@ import { css, html, LitElement } from "lit"; | |||||||
| import { customElement } from "lit/decorators"; | import { customElement } from "lit/decorators"; | ||||||
| import { ifDefined } from "lit/directives/if-defined"; | import { ifDefined } from "lit/directives/if-defined"; | ||||||
| import { repeat } from "lit/directives/repeat"; | import { repeat } from "lit/directives/repeat"; | ||||||
| import "../../../../src/components/ha-control-button"; |  | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
| import "../../../../src/components/ha-svg-icon"; | import "../../../../src/components/ha-control-button"; | ||||||
| import "../../../../src/components/ha-control-button-group"; | import "../../../../src/components/ha-control-button-group"; | ||||||
|  | import "../../../../src/components/ha-svg-icon"; | ||||||
|  |  | ||||||
| interface Button { | interface Button { | ||||||
|   label: string; |   label: string; | ||||||
| @@ -156,17 +156,17 @@ export class DemoHaBarButton extends LitElement { | |||||||
|       --control-button-icon-color: var(--primary-color); |       --control-button-icon-color: var(--primary-color); | ||||||
|       --control-button-background-color: var(--primary-color); |       --control-button-background-color: var(--primary-color); | ||||||
|       --control-button-background-opacity: 0.2; |       --control-button-background-opacity: 0.2; | ||||||
|       --control-button-border-radius: 18px; |       --control-button-border-radius: var(--ha-border-radius-xl); | ||||||
|       height: 100px; |       height: 100px; | ||||||
|       width: 100px; |       width: 100px; | ||||||
|     } |     } | ||||||
|     .custom-group { |     .custom-group { | ||||||
|       --control-button-group-thickness: 100px; |       --control-button-group-thickness: 100px; | ||||||
|       --control-button-group-border-radius: 36px; |       --control-button-group-border-radius: var(--ha-border-radius-6xl); | ||||||
|       --control-button-group-spacing: 20px; |       --control-button-group-spacing: 20px; | ||||||
|     } |     } | ||||||
|     .custom-group ha-control-button { |     .custom-group ha-control-button { | ||||||
|       --control-button-border-radius: 18px; |       --control-button-border-radius: var(--ha-border-radius-xl); | ||||||
|       --mdc-icon-size: 32px; |       --mdc-icon-size: 32px; | ||||||
|     } |     } | ||||||
|     .vertical-buttons { |     .vertical-buttons { | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { LitElement, css, html } from "lit"; | import { LitElement, css, html } from "lit"; | ||||||
| import { customElement, state } from "lit/decorators"; | import { customElement, state } from "lit/decorators"; | ||||||
|  | import { ifDefined } from "lit/directives/if-defined"; | ||||||
|  | import { repeat } from "lit/directives/repeat"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
| import "../../../../src/components/ha-control-number-buttons"; | import "../../../../src/components/ha-control-number-buttons"; | ||||||
| import { repeat } from "lit/directives/repeat"; |  | ||||||
| import { ifDefined } from "lit/directives/if-defined"; |  | ||||||
|  |  | ||||||
| const buttons: { | const buttons: { | ||||||
|   id: string; |   id: string; | ||||||
| @@ -94,7 +94,7 @@ export class DemoHarControlNumberButtons extends LitElement { | |||||||
|       --control-number-buttons-background-color: #2196f3; |       --control-number-buttons-background-color: #2196f3; | ||||||
|       --control-number-buttons-background-opacity: 0.1; |       --control-number-buttons-background-opacity: 0.1; | ||||||
|       --control-number-buttons-thickness: 100px; |       --control-number-buttons-thickness: 100px; | ||||||
|       --control-number-buttons-border-radius: 36px; |       --control-number-buttons-border-radius: var(--ha-border-radius-6xl); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -131,7 +131,7 @@ export class DemoHaControlSelectMenu extends LitElement { | |||||||
|       --control-button-icon-color: var(--primary-color); |       --control-button-icon-color: var(--primary-color); | ||||||
|       --control-button-background-color: var(--primary-color); |       --control-button-background-color: var(--primary-color); | ||||||
|       --control-button-background-opacity: 0.2; |       --control-button-background-opacity: 0.2; | ||||||
|       --control-button-border-radius: 18px; |       --control-button-border-radius: var(--ha-border-radius-xl); | ||||||
|       height: 100px; |       height: 100px; | ||||||
|       width: 100px; |       width: 100px; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -135,7 +135,7 @@ export class DemoHaControlSelect extends LitElement { | |||||||
|                 .options=${options} |                 .options=${options} | ||||||
|                 class=${ifDefined(config.class)} |                 class=${ifDefined(config.class)} | ||||||
|                 @value-changed=${this.handleValueChanged} |                 @value-changed=${this.handleValueChanged} | ||||||
|                 aria-labelledby=${id} |                 .label=${label} | ||||||
|                 ?disabled=${config.disabled} |                 ?disabled=${config.disabled} | ||||||
|               > |               > | ||||||
|               </ha-control-select> |               </ha-control-select> | ||||||
| @@ -156,7 +156,7 @@ export class DemoHaControlSelect extends LitElement { | |||||||
|                   vertical |                   vertical | ||||||
|                   class=${ifDefined(config.class)} |                   class=${ifDefined(config.class)} | ||||||
|                   @value-changed=${this.handleValueChanged} |                   @value-changed=${this.handleValueChanged} | ||||||
|                   aria-labelledby=${id} |                   .label=${label} | ||||||
|                   ?disabled=${config.disabled} |                   ?disabled=${config.disabled} | ||||||
|                 > |                 > | ||||||
|                 </ha-control-select> |                 </ha-control-select> | ||||||
| @@ -187,7 +187,7 @@ export class DemoHaControlSelect extends LitElement { | |||||||
|       --mdc-icon-size: 24px; |       --mdc-icon-size: 24px; | ||||||
|       --control-select-color: var(--state-fan-active-color); |       --control-select-color: var(--state-fan-active-color); | ||||||
|       --control-select-thickness: 130px; |       --control-select-thickness: 130px; | ||||||
|       --control-select-border-radius: 36px; |       --control-select-border-radius: var(--ha-border-radius-6xl); | ||||||
|     } |     } | ||||||
|     .vertical-selects { |     .vertical-selects { | ||||||
|       height: 300px; |       height: 300px; | ||||||
|   | |||||||
| @@ -3,8 +3,8 @@ import { css, html, LitElement } from "lit"; | |||||||
| import { customElement, state } from "lit/decorators"; | import { customElement, state } from "lit/decorators"; | ||||||
| import { ifDefined } from "lit/directives/if-defined"; | import { ifDefined } from "lit/directives/if-defined"; | ||||||
| import { repeat } from "lit/directives/repeat"; | import { repeat } from "lit/directives/repeat"; | ||||||
| import "../../../../src/components/ha-control-slider"; |  | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-control-slider"; | ||||||
|  |  | ||||||
| const sliders: { | const sliders: { | ||||||
|   id: string; |   id: string; | ||||||
| @@ -97,7 +97,7 @@ export class DemoHaBarSlider extends LitElement { | |||||||
|                 class=${ifDefined(config.class)} |                 class=${ifDefined(config.class)} | ||||||
|                 @value-changed=${this.handleValueChanged} |                 @value-changed=${this.handleValueChanged} | ||||||
|                 @slider-moved=${this.handleSliderMoved} |                 @slider-moved=${this.handleSliderMoved} | ||||||
|                 aria-labelledby=${id} |                 .label=${label} | ||||||
|                 .unit=${config.unit} |                 .unit=${config.unit} | ||||||
|               > |               > | ||||||
|               </ha-control-slider> |               </ha-control-slider> | ||||||
| @@ -119,7 +119,7 @@ export class DemoHaBarSlider extends LitElement { | |||||||
|                   class=${ifDefined(config.class)} |                   class=${ifDefined(config.class)} | ||||||
|                   @value-changed=${this.handleValueChanged} |                   @value-changed=${this.handleValueChanged} | ||||||
|                   @slider-moved=${this.handleSliderMoved} |                   @slider-moved=${this.handleSliderMoved} | ||||||
|                   aria-label=${label} |                   .label=${label} | ||||||
|                   .unit=${config.unit} |                   .unit=${config.unit} | ||||||
|                 > |                 > | ||||||
|                 </ha-control-slider> |                 </ha-control-slider> | ||||||
| @@ -151,7 +151,7 @@ export class DemoHaBarSlider extends LitElement { | |||||||
|       --control-slider-background: #ffcf4c; |       --control-slider-background: #ffcf4c; | ||||||
|       --control-slider-background-opacity: 0.2; |       --control-slider-background-opacity: 0.2; | ||||||
|       --control-slider-thickness: 130px; |       --control-slider-thickness: 130px; | ||||||
|       --control-slider-border-radius: 36px; |       --control-slider-border-radius: var(--ha-border-radius-6xl); | ||||||
|     } |     } | ||||||
|     .vertical-sliders { |     .vertical-sliders { | ||||||
|       height: 300px; |       height: 300px; | ||||||
|   | |||||||
| @@ -9,8 +9,8 @@ import { css, html, LitElement } from "lit"; | |||||||
| import { customElement, state } from "lit/decorators"; | import { customElement, state } from "lit/decorators"; | ||||||
| import { ifDefined } from "lit/directives/if-defined"; | import { ifDefined } from "lit/directives/if-defined"; | ||||||
| import { repeat } from "lit/directives/repeat"; | import { repeat } from "lit/directives/repeat"; | ||||||
| import "../../../../src/components/ha-control-switch"; |  | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-control-switch"; | ||||||
|  |  | ||||||
| const switches: { | const switches: { | ||||||
|   id: string; |   id: string; | ||||||
| @@ -63,7 +63,7 @@ export class DemoHaControlSwitch extends LitElement { | |||||||
|                 @change=${this.handleValueChanged} |                 @change=${this.handleValueChanged} | ||||||
|                 .pathOn=${mdiLightbulb} |                 .pathOn=${mdiLightbulb} | ||||||
|                 .pathOff=${mdiLightbulbOff} |                 .pathOff=${mdiLightbulbOff} | ||||||
|                 aria-labelledby=${id} |                 .label=${label} | ||||||
|                 ?disabled=${config.disabled} |                 ?disabled=${config.disabled} | ||||||
|                 ?reversed=${config.reversed} |                 ?reversed=${config.reversed} | ||||||
|               > |               > | ||||||
| @@ -84,7 +84,7 @@ export class DemoHaControlSwitch extends LitElement { | |||||||
|                   vertical |                   vertical | ||||||
|                   class=${ifDefined(config.class)} |                   class=${ifDefined(config.class)} | ||||||
|                   @change=${this.handleValueChanged} |                   @change=${this.handleValueChanged} | ||||||
|                   aria-label=${label} |                   .label=${label} | ||||||
|                   .pathOn=${mdiGarageOpen} |                   .pathOn=${mdiGarageOpen} | ||||||
|                   .pathOff=${mdiGarage} |                   .pathOff=${mdiGarage} | ||||||
|                   ?disabled=${config.disabled} |                   ?disabled=${config.disabled} | ||||||
| @@ -118,7 +118,7 @@ export class DemoHaControlSwitch extends LitElement { | |||||||
|       --control-switch-on-color: var(--green-color); |       --control-switch-on-color: var(--green-color); | ||||||
|       --control-switch-off-color: var(--red-color); |       --control-switch-off-color: var(--red-color); | ||||||
|       --control-switch-thickness: 130px; |       --control-switch-thickness: 130px; | ||||||
|       --control-switch-border-radius: 36px; |       --control-switch-border-radius: var(--ha-border-radius-6xl); | ||||||
|       --control-switch-padding: 6px; |       --control-switch-padding: 6px; | ||||||
|       --mdc-icon-size: 24px; |       --mdc-icon-size: 24px; | ||||||
|     } |     } | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								gallery/src/pages/components/ha-dropdown.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								gallery/src/pages/components/ha-dropdown.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | --- | ||||||
|  | title: Dropdown | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | # Dropdown `<ha-dropdown>` | ||||||
							
								
								
									
										133
									
								
								gallery/src/pages/components/ha-dropdown.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								gallery/src/pages/components/ha-dropdown.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,133 @@ | |||||||
|  | import { | ||||||
|  |   mdiContentCopy, | ||||||
|  |   mdiContentCut, | ||||||
|  |   mdiContentPaste, | ||||||
|  |   mdiDelete, | ||||||
|  | } from "@mdi/js"; | ||||||
|  | import type { TemplateResult } from "lit"; | ||||||
|  | import { css, html, LitElement } from "lit"; | ||||||
|  | import { customElement } from "lit/decorators"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-svg-icon"; | ||||||
|  | import "../../../../src/components/ha-dropdown-item"; | ||||||
|  | import "@home-assistant/webawesome/dist/components/icon/icon"; | ||||||
|  | import "@home-assistant/webawesome/dist/components/button/button"; | ||||||
|  | import "@home-assistant/webawesome/dist/components/dropdown/dropdown"; | ||||||
|  | import "../../../../src/components/ha-dropdown"; | ||||||
|  | import "@home-assistant/webawesome/dist/components/popup/popup"; | ||||||
|  | import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||||
|  | import "../../../../src/components/ha-icon-button"; | ||||||
|  |  | ||||||
|  | @customElement("demo-components-ha-dropdown") | ||||||
|  | export class DemoHaDropdown extends LitElement { | ||||||
|  |   protected render(): TemplateResult { | ||||||
|  |     return html` | ||||||
|  |       ${["light", "dark"].map( | ||||||
|  |         (mode) => html` | ||||||
|  |           <div class=${mode}> | ||||||
|  |             <ha-card header="ha-button in ${mode}"> | ||||||
|  |               <div class="card-content"> | ||||||
|  |                 <ha-dropdown open> | ||||||
|  |                   <ha-button slot="trigger" with-caret>Dropdown</ha-button> | ||||||
|  |  | ||||||
|  |                   <ha-dropdown-item> | ||||||
|  |                     <ha-svg-icon | ||||||
|  |                       .path=${mdiContentCut} | ||||||
|  |                       slot="icon" | ||||||
|  |                     ></ha-svg-icon> | ||||||
|  |                     Cut | ||||||
|  |                   </ha-dropdown-item> | ||||||
|  |                   <ha-dropdown-item> | ||||||
|  |                     <ha-svg-icon | ||||||
|  |                       .path=${mdiContentCopy} | ||||||
|  |                       slot="icon" | ||||||
|  |                     ></ha-svg-icon> | ||||||
|  |                     Copy | ||||||
|  |                   </ha-dropdown-item> | ||||||
|  |                   <ha-dropdown-item disabled> | ||||||
|  |                     <ha-svg-icon | ||||||
|  |                       .path=${mdiContentPaste} | ||||||
|  |                       slot="icon" | ||||||
|  |                     ></ha-svg-icon> | ||||||
|  |                     Paste | ||||||
|  |                   </ha-dropdown-item> | ||||||
|  |                   <ha-dropdown-item> | ||||||
|  |                     Show images | ||||||
|  |                     <ha-dropdown-item slot="submenu" value="show-all-images" | ||||||
|  |                       >Show All Images</ha-dropdown-item | ||||||
|  |                     > | ||||||
|  |                     <ha-dropdown-item slot="submenu" value="show-thumbnails" | ||||||
|  |                       >Show Thumbnails</ha-dropdown-item | ||||||
|  |                     > | ||||||
|  |                   </ha-dropdown-item> | ||||||
|  |                   <ha-dropdown-item type="checkbox" checked | ||||||
|  |                     >Emoji Shortcuts</ha-dropdown-item | ||||||
|  |                   > | ||||||
|  |                   <ha-dropdown-item type="checkbox" checked | ||||||
|  |                     >Word Wrap</ha-dropdown-item | ||||||
|  |                   > | ||||||
|  |                   <ha-dropdown-item variant="danger"> | ||||||
|  |                     <ha-svg-icon .path=${mdiDelete} slot="icon"></ha-svg-icon> | ||||||
|  |                     Delete | ||||||
|  |                   </ha-dropdown-item> | ||||||
|  |                 </ha-dropdown> | ||||||
|  |               </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 styles = css` | ||||||
|  |     :host { | ||||||
|  |       display: flex; | ||||||
|  |       justify-content: center; | ||||||
|  |     } | ||||||
|  |     .dark, | ||||||
|  |     .light { | ||||||
|  |       display: block; | ||||||
|  |       background-color: var(--primary-background-color); | ||||||
|  |       padding: 0 50px; | ||||||
|  |     } | ||||||
|  |     .button { | ||||||
|  |       padding: unset; | ||||||
|  |     } | ||||||
|  |     ha-card { | ||||||
|  |       margin: 24px auto; | ||||||
|  |     } | ||||||
|  |     .card-content { | ||||||
|  |       display: flex; | ||||||
|  |       flex-direction: column; | ||||||
|  |       gap: 24px; | ||||||
|  |     } | ||||||
|  |     .card-content div { | ||||||
|  |       display: flex; | ||||||
|  |       gap: 8px; | ||||||
|  |     } | ||||||
|  |   `; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-components-ha-dropdown": DemoHaDropdown; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,5 +1,4 @@ | |||||||
| /* eslint-disable lit/no-template-arrow */ | /* eslint-disable lit/no-template-arrow */ | ||||||
| import "@material/mwc-button"; |  | ||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { html, LitElement } from "lit"; | import { html, LitElement } from "lit"; | ||||||
| import { customElement, state } from "lit/decorators"; | import { customElement, state } from "lit/decorators"; | ||||||
|   | |||||||
							
								
								
									
										37
									
								
								gallery/src/pages/components/ha-marquee-text.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								gallery/src/pages/components/ha-marquee-text.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | |||||||
|  | --- | ||||||
|  | title: Marquee Text | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | # Marquee Text `<ha-marquee-text>` | ||||||
|  |  | ||||||
|  | Marquee text component scrolls text horizontally if it overflows its container. It supports pausing on hover and customizable speed and pause duration. | ||||||
|  |  | ||||||
|  | ## Implementation | ||||||
|  |  | ||||||
|  | ### Example Usage | ||||||
|  |  | ||||||
|  | <ha-marquee-text style="width: 200px;"> | ||||||
|  |     This is a long text that will scroll horizontally if it overflows the container. | ||||||
|  | </ha-marquee-text> | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <ha-marquee-text style="width: 200px;"> | ||||||
|  |   This is a long text that will scroll horizontally if it overflows the | ||||||
|  |   container. | ||||||
|  | </ha-marquee-text> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### API | ||||||
|  |  | ||||||
|  | **Slots** | ||||||
|  |  | ||||||
|  | - default slot: The text content to be displayed and scrolled. | ||||||
|  |   - no default | ||||||
|  |  | ||||||
|  | **Properties/Attributes** | ||||||
|  |  | ||||||
|  | | Name           | Type    | Default | Description                                                                  | | ||||||
|  | | -------------- | ------- | ------- | ---------------------------------------------------------------------------- | | ||||||
|  | | speed          | number  | `15`    | The speed of the scrolling animation. Higher values result in faster scroll. | | ||||||
|  | | pause-on-hover | boolean | `true`  | Whether to pause the scrolling animation when                                | | ||||||
|  | | pause-duration | number  | `1000`  | The delay in milliseconds before the scrolling animation starts/restarts.    | | ||||||
							
								
								
									
										25
									
								
								gallery/src/pages/components/ha-marquee-text.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								gallery/src/pages/components/ha-marquee-text.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | |||||||
|  | import { css, LitElement } from "lit"; | ||||||
|  | import { customElement } from "lit/decorators"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-marquee-text"; | ||||||
|  |  | ||||||
|  | @customElement("demo-components-ha-marquee-text") | ||||||
|  | export class DemoHaMarqueeText extends LitElement { | ||||||
|  |   static styles = css` | ||||||
|  |     ha-card { | ||||||
|  |       max-width: 600px; | ||||||
|  |       margin: 24px auto; | ||||||
|  |     } | ||||||
|  |     .card-content { | ||||||
|  |       display: flex; | ||||||
|  |       flex-direction: column; | ||||||
|  |       align-items: flex-start; | ||||||
|  |     } | ||||||
|  |   `; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-components-ha-marquee-text": DemoHaMarqueeText; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										32
									
								
								gallery/src/pages/components/ha-progress-button.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								gallery/src/pages/components/ha-progress-button.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | |||||||
|  | --- | ||||||
|  | title: Progress Button | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   .wrapper { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 24px; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  | # Progress Button `<ha-progress-button>` | ||||||
|  |  | ||||||
|  | ### API | ||||||
|  |  | ||||||
|  | This component is a wrapper around `<ha-button>` that adds support for showing progress | ||||||
|  |  | ||||||
|  | **Slots** | ||||||
|  |  | ||||||
|  | - default slot: Label of the button | ||||||
|  |   ` - no default | ||||||
|  |  | ||||||
|  | **Properties/Attributes** | ||||||
|  |  | ||||||
|  | | Name       | Type                                           | Default   | Description                                        | | ||||||
|  | | ---------- | ---------------------------------------------- | --------- | -------------------------------------------------- | | ||||||
|  | | label      | string                                         | "accent"  | Sets the button label.                             | | ||||||
|  | | disabled   | Boolean                                        | false     | Disables the button if true.                       | | ||||||
|  | | progress   | Boolean                                        | false     | Shows a progress indicator on the button.          | | ||||||
|  | | appearance | "accent"/"filled"/"plain"                      | "accent"  | Sets the button appearance.                        | | ||||||
|  | | variants   | "brand"/"danger"/"neutral"/"warning"/"success" | "brand"   | Sets the button color variant. "brand" is default. | | ||||||
|  | | iconPath   | string                                         | undefined | Sets the icon path for the button.                 | | ||||||
							
								
								
									
										139
									
								
								gallery/src/pages/components/ha-progress-button.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								gallery/src/pages/components/ha-progress-button.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,139 @@ | |||||||
|  | import type { TemplateResult } from "lit"; | ||||||
|  | import { css, html, LitElement } from "lit"; | ||||||
|  | import { customElement } from "lit/decorators"; | ||||||
|  | import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||||
|  | import "../../../../src/components/buttons/ha-progress-button"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-svg-icon"; | ||||||
|  | import { mdiHomeAssistant } from "../../../../src/resources/home-assistant-logo-svg"; | ||||||
|  |  | ||||||
|  | @customElement("demo-components-ha-progress-button") | ||||||
|  | export class DemoHaProgressButton extends LitElement { | ||||||
|  |   protected render(): TemplateResult { | ||||||
|  |     return html` | ||||||
|  |       ${["light", "dark"].map( | ||||||
|  |         (mode) => html` | ||||||
|  |           <div class=${mode}> | ||||||
|  |             <ha-card header="ha-progress-button in ${mode}"> | ||||||
|  |               <div class="card-content"> | ||||||
|  |                 <ha-progress-button @click=${this._clickedSuccess}> | ||||||
|  |                   Success | ||||||
|  |                 </ha-progress-button> | ||||||
|  |                 <ha-progress-button @click=${this._clickedFail}> | ||||||
|  |                   Fail | ||||||
|  |                 </ha-progress-button> | ||||||
|  |                 <ha-progress-button size="small" @click=${this._clickedSuccess}> | ||||||
|  |                   small | ||||||
|  |                 </ha-progress-button> | ||||||
|  |                 <ha-progress-button | ||||||
|  |                   appearance="filled" | ||||||
|  |                   @click=${this._clickedSuccess} | ||||||
|  |                 > | ||||||
|  |                   filled | ||||||
|  |                 </ha-progress-button> | ||||||
|  |                 <ha-progress-button | ||||||
|  |                   appearance="plain" | ||||||
|  |                   @click=${this._clickedSuccess} | ||||||
|  |                 > | ||||||
|  |                   plain | ||||||
|  |                 </ha-progress-button> | ||||||
|  |                 <ha-progress-button | ||||||
|  |                   variant="warning" | ||||||
|  |                   @click=${this._clickedSuccess} | ||||||
|  |                 > | ||||||
|  |                   warning | ||||||
|  |                 </ha-progress-button> | ||||||
|  |                 <ha-progress-button | ||||||
|  |                   variant="neutral" | ||||||
|  |                   @click=${this._clickedSuccess} | ||||||
|  |                   label="with icon" | ||||||
|  |                   .iconPath=${mdiHomeAssistant} | ||||||
|  |                 > | ||||||
|  |                   With Icon | ||||||
|  |                 </ha-progress-button> | ||||||
|  |                 <ha-progress-button progress @click=${this._clickedSuccess}> | ||||||
|  |                   progress | ||||||
|  |                 </ha-progress-button> | ||||||
|  |                 <ha-progress-button disabled @click=${this._clickedSuccess}> | ||||||
|  |                   disabled | ||||||
|  |                 </ha-progress-button> | ||||||
|  |               </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 | ||||||
|  |     ); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private async _clickedSuccess(ev: CustomEvent): Promise<void> { | ||||||
|  |     console.log("Clicked success"); | ||||||
|  |     const button = ev.currentTarget as any; | ||||||
|  |     button.progress = true; | ||||||
|  |  | ||||||
|  |     setTimeout(() => { | ||||||
|  |       button.actionSuccess(); | ||||||
|  |       button.progress = false; | ||||||
|  |     }, 1000); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private async _clickedFail(ev: CustomEvent): Promise<void> { | ||||||
|  |     const button = ev.currentTarget as any; | ||||||
|  |     button.progress = true; | ||||||
|  |  | ||||||
|  |     setTimeout(() => { | ||||||
|  |       button.actionError(); | ||||||
|  |       button.progress = false; | ||||||
|  |     }, 1000); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   static styles = css` | ||||||
|  |     :host { | ||||||
|  |       display: flex; | ||||||
|  |       justify-content: center; | ||||||
|  |     } | ||||||
|  |     .dark, | ||||||
|  |     .light { | ||||||
|  |       display: block; | ||||||
|  |       background-color: var(--primary-background-color); | ||||||
|  |       padding: 0 50px; | ||||||
|  |     } | ||||||
|  |     .button { | ||||||
|  |       padding: unset; | ||||||
|  |     } | ||||||
|  |     ha-card { | ||||||
|  |       margin: 24px auto; | ||||||
|  |     } | ||||||
|  |     .card-content { | ||||||
|  |       display: flex; | ||||||
|  |       flex-direction: column; | ||||||
|  |       gap: var(--ha-space-6); | ||||||
|  |     } | ||||||
|  |     .card-content div { | ||||||
|  |       display: flex; | ||||||
|  |       gap: var(--ha-space-2); | ||||||
|  |     } | ||||||
|  |   `; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-components-ha-progress-button": DemoHaProgressButton; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -131,7 +131,7 @@ export class DemoHaSelectBox extends LitElement { | |||||||
|       --mdc-icon-size: 24px; |       --mdc-icon-size: 24px; | ||||||
|       --control-select-color: var(--state-fan-active-color); |       --control-select-color: var(--state-fan-active-color); | ||||||
|       --control-select-thickness: 130px; |       --control-select-thickness: 130px; | ||||||
|       --control-select-border-radius: 36px; |       --control-select-border-radius: var(--ha-border-radius-6xl); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     p.title { |     p.title { | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement, state } from "lit/decorators"; | import { customElement, state } from "lit/decorators"; | ||||||
|   | |||||||
							
								
								
									
										38
									
								
								gallery/src/pages/components/ha-slider.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								gallery/src/pages/components/ha-slider.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | |||||||
|  | --- | ||||||
|  | title: Slider | ||||||
|  | subtitle: A slider component for selecting a value from a range. | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | <style> | ||||||
|  |   .wrapper { | ||||||
|  |     display: flex; | ||||||
|  |     gap: 24px; | ||||||
|  |   } | ||||||
|  | </style> | ||||||
|  |  | ||||||
|  | # Slider `<ha-slider>` | ||||||
|  |  | ||||||
|  | ## Implementation | ||||||
|  |  | ||||||
|  | ### Example Usage | ||||||
|  |  | ||||||
|  | <div class="wrapper"> | ||||||
|  |   <ha-slider size="small" with-markers min="0" max="8" value="4"></ha-slider> | ||||||
|  |   <ha-slider size="medium"></ha-slider> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <ha-slider size="small" with-markers min="0" max="8" value="4"></ha-slider> | ||||||
|  | <ha-slider size="medium"></ha-slider> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### API | ||||||
|  |  | ||||||
|  | This component is based on the webawesome slider component. | ||||||
|  | Check the [webawesome documentation](https://webawesome.com/docs/components/slider/) for more details. | ||||||
|  |  | ||||||
|  | **CSS Custom Properties** | ||||||
|  |  | ||||||
|  | - `--ha-slider-track-size` - Height of the slider track. Defaults to `4px`. | ||||||
|  | - `--ha-slider-thumb-color` - Color of the slider thumb. Defaults to `var(--primary-color)`. | ||||||
|  | - `--ha-slider-indicator-color` - Color of the filled portion of the slider track. Defaults to `var(--primary-color)`. | ||||||
							
								
								
									
										100
									
								
								gallery/src/pages/components/ha-slider.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								gallery/src/pages/components/ha-slider.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | |||||||
|  | import type { TemplateResult } from "lit"; | ||||||
|  | import { css, html, LitElement } from "lit"; | ||||||
|  | import { customElement, property } from "lit/decorators"; | ||||||
|  | import { applyThemesOnElement } from "../../../../src/common/dom/apply_themes_on_element"; | ||||||
|  | import "../../../../src/components/ha-bar"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-spinner"; | ||||||
|  | import "../../../../src/components/ha-slider"; | ||||||
|  | import type { HomeAssistant } from "../../../../src/types"; | ||||||
|  |  | ||||||
|  | @customElement("demo-components-ha-slider") | ||||||
|  | export class DemoHaSlider extends LitElement { | ||||||
|  |   @property({ attribute: false }) hass!: HomeAssistant; | ||||||
|  |  | ||||||
|  |   protected render(): TemplateResult { | ||||||
|  |     return html` | ||||||
|  |       ${["light", "dark"].map( | ||||||
|  |         (mode) => html` | ||||||
|  |           <div class=${mode}> | ||||||
|  |             <ha-card header="ha-slider ${mode} demo"> | ||||||
|  |               <div class="card-content"> | ||||||
|  |                 <span>Default (disabled)</span> | ||||||
|  |                 <ha-slider | ||||||
|  |                   disabled | ||||||
|  |                   min="0" | ||||||
|  |                   max="8" | ||||||
|  |                   value="4" | ||||||
|  |                   with-markers | ||||||
|  |                 ></ha-slider> | ||||||
|  |                 <span>Small</span> | ||||||
|  |                 <ha-slider | ||||||
|  |                   size="small" | ||||||
|  |                   min="0" | ||||||
|  |                   max="8" | ||||||
|  |                   value="4" | ||||||
|  |                   with-markers | ||||||
|  |                 ></ha-slider> | ||||||
|  |                 <span>Medium</span> | ||||||
|  |                 <ha-slider | ||||||
|  |                   size="medium" | ||||||
|  |                   min="0" | ||||||
|  |                   max="8" | ||||||
|  |                   value="4" | ||||||
|  |                   with-markers | ||||||
|  |                 ></ha-slider> | ||||||
|  |               </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 styles = css` | ||||||
|  |     :host { | ||||||
|  |       display: flex; | ||||||
|  |       justify-content: center; | ||||||
|  |     } | ||||||
|  |     .dark, | ||||||
|  |     .light { | ||||||
|  |       display: block; | ||||||
|  |       background-color: var(--primary-background-color); | ||||||
|  |       padding: 0 50px; | ||||||
|  |       margin: 16px; | ||||||
|  |       border-radius: var(--ha-border-radius-md); | ||||||
|  |     } | ||||||
|  |     ha-card { | ||||||
|  |       margin: 24px auto; | ||||||
|  |     } | ||||||
|  |     .card-content { | ||||||
|  |       display: flex; | ||||||
|  |       flex-direction: column; | ||||||
|  |       align-items: center; | ||||||
|  |       gap: var(--ha-space-6); | ||||||
|  |     } | ||||||
|  |   `; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-components-ha-slider": DemoHaSlider; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -61,7 +61,7 @@ export class DemoHaSpinner extends LitElement { | |||||||
|       background-color: var(--primary-background-color); |       background-color: var(--primary-background-color); | ||||||
|       padding: 0 50px; |       padding: 0 50px; | ||||||
|       margin: 16px; |       margin: 16px; | ||||||
|       border-radius: 8px; |       border-radius: var(--ha-border-radius-md); | ||||||
|     } |     } | ||||||
|     ha-card { |     ha-card { | ||||||
|       margin: 24px auto; |       margin: 24px auto; | ||||||
| @@ -70,7 +70,7 @@ export class DemoHaSpinner extends LitElement { | |||||||
|       display: flex; |       display: flex; | ||||||
|       flex-direction: column; |       flex-direction: column; | ||||||
|       align-items: center; |       align-items: center; | ||||||
|       gap: 24px; |       gap: var(--ha-space-6); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,21 +6,23 @@ A tooltip's target is its _first child element_, so you should only wrap one ele | |||||||
|  |  | ||||||
| Tooltips use `display: contents` so they won't interfere with how elements are positioned in a flex or grid layout. | Tooltips use `display: contents` so they won't interfere with how elements are positioned in a flex or grid layout. | ||||||
|  |  | ||||||
| <ha-tooltip content="This is a tooltip"> | <ha-button id="hover">Hover Me</ha-button> | ||||||
|   <ha-button>Hover Me</ha-button> | <ha-tooltip for="hover"> | ||||||
|  | This is a tooltip | ||||||
| </ha-tooltip> | </ha-tooltip> | ||||||
|  |  | ||||||
| ``` | ``` | ||||||
| <ha-tooltip content="This is a tooltip"> | <ha-button id="hover">Hover Me</ha-button> | ||||||
|   <ha-button>Hover Me</ha-button> | <ha-tooltip for="hover"> | ||||||
|  | This is a tooltip | ||||||
| </ha-tooltip> | </ha-tooltip> | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| ## Documentation | ## Documentation | ||||||
|  |  | ||||||
| This element is based on shoelace `sl-tooltip` it only sets some css tokens and has a custom show/hide animation. | This element is based on webawesome `wa-tooltip` it only sets some css tokens and has a custom show/hide animation. | ||||||
|  |  | ||||||
| <a href="https://shoelace.style/components/tooltip" target="_blank" rel="noopener noreferrer">Shoelace documentation</a> | <a href="https://webawesome.com/docs/components/tooltip/" target="_blank" rel="noopener noreferrer">Webawesome documentation</a> | ||||||
|  |  | ||||||
| ### HA style tokens | ### HA style tokens | ||||||
|  |  | ||||||
| @@ -28,7 +30,7 @@ In your theme settings use this without the prefixed `--`. | |||||||
|  |  | ||||||
| - `--ha-tooltip-border-radius` (Default: 4px) | - `--ha-tooltip-border-radius` (Default: 4px) | ||||||
| - `--ha-tooltip-arrow-size` (Default: 8px) | - `--ha-tooltip-arrow-size` (Default: 8px) | ||||||
| - `--sl-tooltip-font-family` (Default: `var(--ha-font-family-body)`) | - `--wa-tooltip-font-family` (Default: `var(--ha-font-family-body)`) | ||||||
| - `--ha-tooltip-font-size` (Default: `var(--ha-font-size-s)`) | - `--ha-tooltip-font-size` (Default: `var(--ha-font-size-s)`) | ||||||
| - `--sl-tooltip-font-weight` (Default: `var(--ha-font-weight-normal)`) | - `--wa-tooltip-font-weight` (Default: `var(--ha-font-weight-normal)`) | ||||||
| - `--sl-tooltip-line-height` (Default: `var(--ha-line-height-condensed)`) | - `--wa-tooltip-line-height` (Default: `var(--ha-line-height-condensed)`) | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								gallery/src/pages/components/ha-wa-dialog.markdown
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/components/ha-wa-dialog.markdown
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | --- | ||||||
|  | title: Dialog (ha-wa-dialog) | ||||||
|  | --- | ||||||
							
								
								
									
										523
									
								
								gallery/src/pages/components/ha-wa-dialog.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										523
									
								
								gallery/src/pages/components/ha-wa-dialog.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,523 @@ | |||||||
|  | import { css, html, LitElement } from "lit"; | ||||||
|  | import { customElement, state } from "lit/decorators"; | ||||||
|  | import { mdiCog, mdiHelp } from "@mdi/js"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/components/ha-dialog-footer"; | ||||||
|  | import "../../../../src/components/ha-form/ha-form"; | ||||||
|  | import "../../../../src/components/ha-icon-button"; | ||||||
|  | import "../../../../src/components/ha-wa-dialog"; | ||||||
|  | import type { HaFormSchema } from "../../../../src/components/ha-form/types"; | ||||||
|  |  | ||||||
|  | const SCHEMA: HaFormSchema[] = [ | ||||||
|  |   { type: "string", name: "Name", default: "", autofocus: true }, | ||||||
|  |   { type: "string", name: "Email", default: "" }, | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | type DialogType = | ||||||
|  |   | false | ||||||
|  |   | "basic" | ||||||
|  |   | "basic-subtitle-below" | ||||||
|  |   | "basic-subtitle-above" | ||||||
|  |   | "form" | ||||||
|  |   | "actions"; | ||||||
|  |  | ||||||
|  | @customElement("demo-components-ha-wa-dialog") | ||||||
|  | export class DemoHaWaDialog extends LitElement { | ||||||
|  |   @state() private _openDialog: DialogType = false; | ||||||
|  |  | ||||||
|  |   protected render() { | ||||||
|  |     return html` | ||||||
|  |       <div class="content"> | ||||||
|  |         <h1>Dialog <code><ha-wa-dialog></code></h1> | ||||||
|  |  | ||||||
|  |         <p class="subtitle">Dialog component built with WebAwesome.</p> | ||||||
|  |  | ||||||
|  |         <h2>Demos</h2> | ||||||
|  |  | ||||||
|  |         <div class="buttons"> | ||||||
|  |           <ha-button @click=${this._handleOpenDialog("basic")} | ||||||
|  |             >Basic dialog</ha-button | ||||||
|  |           > | ||||||
|  |           <ha-button @click=${this._handleOpenDialog("basic-subtitle-below")} | ||||||
|  |             >Basic dialog with subtitle below</ha-button | ||||||
|  |           > | ||||||
|  |           <ha-button @click=${this._handleOpenDialog("basic-subtitle-above")} | ||||||
|  |             >Basic dialog with subtitle above</ha-button | ||||||
|  |           > | ||||||
|  |           <ha-button @click=${this._handleOpenDialog("form")} | ||||||
|  |             >Dialog with form</ha-button | ||||||
|  |           > | ||||||
|  |           <ha-button @click=${this._handleOpenDialog("actions")} | ||||||
|  |             >Dialog with actions</ha-button | ||||||
|  |           > | ||||||
|  |         </div> | ||||||
|  |  | ||||||
|  |         <ha-wa-dialog | ||||||
|  |           .open=${this._openDialog === "basic"} | ||||||
|  |           header-title="Basic dialog" | ||||||
|  |           @closed=${this._handleClosed} | ||||||
|  |         > | ||||||
|  |           <div>Dialog content</div> | ||||||
|  |         </ha-wa-dialog> | ||||||
|  |  | ||||||
|  |         <ha-wa-dialog | ||||||
|  |           .open=${this._openDialog === "basic-subtitle-below"} | ||||||
|  |           header-title="Basic dialog with subtitle" | ||||||
|  |           header-subtitle="This is a basic dialog with a subtitle below" | ||||||
|  |           @closed=${this._handleClosed} | ||||||
|  |         > | ||||||
|  |           <div>Dialog content</div> | ||||||
|  |         </ha-wa-dialog> | ||||||
|  |  | ||||||
|  |         <ha-wa-dialog | ||||||
|  |           .open=${this._openDialog === "basic-subtitle-above"} | ||||||
|  |           header-title="Dialog with subtitle above" | ||||||
|  |           header-subtitle="This is a basic dialog with a subtitle above" | ||||||
|  |           header-subtitle-position="above" | ||||||
|  |           @closed=${this._handleClosed} | ||||||
|  |         > | ||||||
|  |           <div>Dialog content</div> | ||||||
|  |         </ha-wa-dialog> | ||||||
|  |  | ||||||
|  |         <ha-wa-dialog | ||||||
|  |           .open=${this._openDialog === "form"} | ||||||
|  |           header-title="Dialog with form" | ||||||
|  |           header-subtitle="This is a dialog with a form and a footer" | ||||||
|  |           prevent-scrim-close | ||||||
|  |           @closed=${this._handleClosed} | ||||||
|  |         > | ||||||
|  |           <ha-form autofocus .schema=${SCHEMA}></ha-form> | ||||||
|  |           <ha-dialog-footer slot="footer"> | ||||||
|  |             <ha-button | ||||||
|  |               data-dialog="close" | ||||||
|  |               slot="secondaryAction" | ||||||
|  |               variant="plain" | ||||||
|  |               >Cancel</ha-button | ||||||
|  |             > | ||||||
|  |             <ha-button data-dialog="close" slot="primaryAction" variant="accent" | ||||||
|  |               >Submit</ha-button | ||||||
|  |             > | ||||||
|  |           </ha-dialog-footer> | ||||||
|  |         </ha-wa-dialog> | ||||||
|  |  | ||||||
|  |         <ha-wa-dialog | ||||||
|  |           .open=${this._openDialog === "actions"} | ||||||
|  |           header-title="Dialog with actions" | ||||||
|  |           header-subtitle="This is a dialog with header actions" | ||||||
|  |           @closed=${this._handleClosed} | ||||||
|  |         > | ||||||
|  |           <div slot="headerActionItems"> | ||||||
|  |             <ha-icon-button label="Settings" path=${mdiCog}></ha-icon-button> | ||||||
|  |             <ha-icon-button label="Help" path=${mdiHelp}></ha-icon-button> | ||||||
|  |           </div> | ||||||
|  |  | ||||||
|  |           <div>Dialog content</div> | ||||||
|  |         </ha-wa-dialog> | ||||||
|  |  | ||||||
|  |         <h2>Design</h2> | ||||||
|  |  | ||||||
|  |         <h3>Width</h3> | ||||||
|  |  | ||||||
|  |         <p>There are multiple widths available for the dialog.</p> | ||||||
|  |  | ||||||
|  |         <table> | ||||||
|  |           <thead> | ||||||
|  |             <tr> | ||||||
|  |               <th>Name</th> | ||||||
|  |               <th>Value</th> | ||||||
|  |             </tr> | ||||||
|  |           </thead> | ||||||
|  |           <tbody> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>small</code></td> | ||||||
|  |               <td><code>min(320px, var(--full-width))</code></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>medium</code></td> | ||||||
|  |               <td><code>min(580px, var(--full-width))</code></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>large</code></td> | ||||||
|  |               <td><code>min(720px, var(--full-width))</code></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>full</code></td> | ||||||
|  |               <td><code>var(--full-width)</code></td> | ||||||
|  |             </tr> | ||||||
|  |           </tbody> | ||||||
|  |         </table> | ||||||
|  |  | ||||||
|  |         <p> | ||||||
|  |           <code>--full-width</code> is calculated based on the available width | ||||||
|  |           of the screen. 95vw is the maximum width of the dialog on a large | ||||||
|  |           screen, while on a small screen it is 100vw minus the safe area | ||||||
|  |           insets. | ||||||
|  |         </p> | ||||||
|  |  | ||||||
|  |         <p>Dialogs have a default width of <code>medium</code>.</p> | ||||||
|  |  | ||||||
|  |         <h3>Prevent scrim close</h3> | ||||||
|  |  | ||||||
|  |         <p> | ||||||
|  |           You can prevent the dialog from being closed by clicking the | ||||||
|  |           scrim/overlay. This is allowed by default. | ||||||
|  |         </p> | ||||||
|  |  | ||||||
|  |         <h3>Header</h3> | ||||||
|  |  | ||||||
|  |         <p>The header contains a title, a subtitle and action items.</p> | ||||||
|  |  | ||||||
|  |         <table> | ||||||
|  |           <thead> | ||||||
|  |             <tr> | ||||||
|  |               <th>Slot</th> | ||||||
|  |               <th>Description</th> | ||||||
|  |             </tr> | ||||||
|  |           </thead> | ||||||
|  |           <tbody> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>header</code></td> | ||||||
|  |               <td>The entire header area.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>headerTitle</code></td> | ||||||
|  |               <td>The header title text.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>headerSubtitle</code></td> | ||||||
|  |               <td>The header subtitle text.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>headerActionItems</code></td> | ||||||
|  |               <td>The header action items.</td> | ||||||
|  |             </tr> | ||||||
|  |           </tbody> | ||||||
|  |         </table> | ||||||
|  |  | ||||||
|  |         <h4>Header title</h4> | ||||||
|  |  | ||||||
|  |         <p>The header title is a text string.</p> | ||||||
|  |  | ||||||
|  |         <h4>Header subtitle</h4> | ||||||
|  |  | ||||||
|  |         <p>The header subtitle is a text string.</p> | ||||||
|  |  | ||||||
|  |         <h4>Header action items</h4> | ||||||
|  |  | ||||||
|  |         <p> | ||||||
|  |           The header action items usually containing icon buttons and/or menu | ||||||
|  |           buttons. | ||||||
|  |         </p> | ||||||
|  |  | ||||||
|  |         <h3>Body</h3> | ||||||
|  |  | ||||||
|  |         <p>The body is the content of the dialog.</p> | ||||||
|  |  | ||||||
|  |         <h3>Footer</h3> | ||||||
|  |  | ||||||
|  |         <p>The footer is the footer of the dialog.</p> | ||||||
|  |  | ||||||
|  |         <p> | ||||||
|  |           It is recommended to use the <code>ha-dialog-footer</code> component | ||||||
|  |           for the footer and to style the buttons inside the footer as so: | ||||||
|  |         </p> | ||||||
|  |  | ||||||
|  |         <table> | ||||||
|  |           <thead> | ||||||
|  |             <tr> | ||||||
|  |               <th>Slot</th> | ||||||
|  |               <th>Description</th> | ||||||
|  |               <th>Variant to use</th> | ||||||
|  |             </tr> | ||||||
|  |           </thead> | ||||||
|  |           <tbody> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>secondaryAction</code></td> | ||||||
|  |               <td>The secondary action button(s).</td> | ||||||
|  |               <td><code>plain</code></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>primaryAction</code></td> | ||||||
|  |               <td>The primary action button(s).</td> | ||||||
|  |               <td><code>accent</code></td> | ||||||
|  |             </tr> | ||||||
|  |           </tbody> | ||||||
|  |         </table> | ||||||
|  |  | ||||||
|  |         <h2>Implementation</h2> | ||||||
|  |  | ||||||
|  |         <h3>Example Usage</h3> | ||||||
|  |  | ||||||
|  |         <pre><code><ha-wa-dialog | ||||||
|  |   open | ||||||
|  |   header-title="Dialog title" | ||||||
|  |   header-subtitle="Dialog subtitle" | ||||||
|  |   prevent-scrim-close | ||||||
|  | > | ||||||
|  |   <div slot="headerActionItems"> | ||||||
|  |     <ha-icon-button label="Settings" path="mdiCog"></ha-icon-button> | ||||||
|  |     <ha-icon-button label="Help" path="mdiHelp"></ha-icon-button> | ||||||
|  |   </div> | ||||||
|  |   <div>Dialog content</div> | ||||||
|  |   <ha-dialog-footer slot="footer"> | ||||||
|  |     <ha-button data-dialog="close" slot="secondaryAction" variant="plain" | ||||||
|  |       >Cancel</ha-button | ||||||
|  |     > | ||||||
|  |     <ha-button slot="primaryAction" variant="accent">Submit</ha-button> | ||||||
|  |   </ha-dialog-footer> | ||||||
|  | </ha-wa-dialog></code></pre> | ||||||
|  |  | ||||||
|  |         <h3>API</h3> | ||||||
|  |  | ||||||
|  |         <p> | ||||||
|  |           This component is based on the webawesome dialog component. Check the | ||||||
|  |           <a | ||||||
|  |             href="https://webawesome.com/docs/components/dialog/" | ||||||
|  |             target="_blank" | ||||||
|  |             rel="noopener noreferrer" | ||||||
|  |             >webawesome documentation</a | ||||||
|  |           > | ||||||
|  |           for more details. | ||||||
|  |         </p> | ||||||
|  |  | ||||||
|  |         <h4>Attributes</h4> | ||||||
|  |  | ||||||
|  |         <table> | ||||||
|  |           <thead> | ||||||
|  |             <tr> | ||||||
|  |               <th>Attribute</th> | ||||||
|  |               <th>Description</th> | ||||||
|  |               <th>Default</th> | ||||||
|  |               <th>Options</th> | ||||||
|  |             </tr> | ||||||
|  |           </thead> | ||||||
|  |           <tbody> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>open</code></td> | ||||||
|  |               <td>Controls the dialog open state.</td> | ||||||
|  |               <td><code>false</code></td> | ||||||
|  |               <td><code>false</code>, <code>true</code></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>width</code></td> | ||||||
|  |               <td>Preferred dialog width preset.</td> | ||||||
|  |               <td><code>medium</code></td> | ||||||
|  |               <td> | ||||||
|  |                 <code>small</code>, <code>medium</code>, <code>large</code>, | ||||||
|  |                 <code>full</code> | ||||||
|  |               </td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>prevent-scrim-close</code></td> | ||||||
|  |               <td> | ||||||
|  |                 Prevents closing the dialog by clicking the scrim/overlay. | ||||||
|  |               </td> | ||||||
|  |               <td><code>false</code></td> | ||||||
|  |               <td><code>true</code></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>header-title</code></td> | ||||||
|  |               <td>Header title text when no custom title slot is provided.</td> | ||||||
|  |               <td></td> | ||||||
|  |               <td></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>header-subtitle</code></td> | ||||||
|  |               <td> | ||||||
|  |                 Header subtitle text when no custom subtitle slot is provided. | ||||||
|  |               </td> | ||||||
|  |               <td></td> | ||||||
|  |               <td></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>header-subtitle-position</code></td> | ||||||
|  |               <td>Position of the subtitle relative to the title.</td> | ||||||
|  |               <td><code>below</code></td> | ||||||
|  |               <td><code>above</code>, <code>below</code></td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>flexcontent</code></td> | ||||||
|  |               <td> | ||||||
|  |                 Makes the dialog body a flex container for flexible layouts. | ||||||
|  |               </td> | ||||||
|  |               <td><code>false</code></td> | ||||||
|  |               <td><code>false</code>, <code>true</code></td> | ||||||
|  |             </tr> | ||||||
|  |           </tbody> | ||||||
|  |         </table> | ||||||
|  |  | ||||||
|  |         <h4>CSS Custom Properties</h4> | ||||||
|  |  | ||||||
|  |         <table> | ||||||
|  |           <thead> | ||||||
|  |             <tr> | ||||||
|  |               <th>CSS Property</th> | ||||||
|  |               <th>Description</th> | ||||||
|  |             </tr> | ||||||
|  |           </thead> | ||||||
|  |           <tbody> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>--dialog-content-padding</code></td> | ||||||
|  |               <td>Padding for dialog content sections.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>--ha-dialog-show-duration</code></td> | ||||||
|  |               <td>Show animation duration.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>--ha-dialog-hide-duration</code></td> | ||||||
|  |               <td>Hide animation duration.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>--ha-dialog-surface-background</code></td> | ||||||
|  |               <td>Dialog background color.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>--ha-dialog-border-radius</code></td> | ||||||
|  |               <td>Border radius of the dialog surface.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>--dialog-z-index</code></td> | ||||||
|  |               <td>Z-index for the dialog.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>--dialog-surface-position</code></td> | ||||||
|  |               <td>CSS position of the dialog surface.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>--dialog-surface-margin-top</code></td> | ||||||
|  |               <td>Top margin for the dialog surface.</td> | ||||||
|  |             </tr> | ||||||
|  |           </tbody> | ||||||
|  |         </table> | ||||||
|  |  | ||||||
|  |         <h4>Events</h4> | ||||||
|  |  | ||||||
|  |         <table> | ||||||
|  |           <thead> | ||||||
|  |             <tr> | ||||||
|  |               <th>Event</th> | ||||||
|  |               <th>Description</th> | ||||||
|  |             </tr> | ||||||
|  |           </thead> | ||||||
|  |           <tbody> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>opened</code></td> | ||||||
|  |               <td>Fired when the dialog is shown.</td> | ||||||
|  |             </tr> | ||||||
|  |             <tr> | ||||||
|  |               <td><code>closed</code></td> | ||||||
|  |               <td>Fired after the dialog is hidden.</td> | ||||||
|  |             </tr> | ||||||
|  |           </tbody> | ||||||
|  |         </table> | ||||||
|  |       </div> | ||||||
|  |     `; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _handleOpenDialog = (dialog: DialogType) => () => { | ||||||
|  |     this._openDialog = dialog; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   private _handleClosed = () => { | ||||||
|  |     this._openDialog = false; | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   static styles = [ | ||||||
|  |     css` | ||||||
|  |       :host { | ||||||
|  |         display: block; | ||||||
|  |         padding: var(--ha-space-4); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       .content { | ||||||
|  |         max-width: 1000px; | ||||||
|  |         margin: 0 auto; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       h1 { | ||||||
|  |         margin-top: 0; | ||||||
|  |         margin-bottom: var(--ha-space-2); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       h2 { | ||||||
|  |         margin-top: var(--ha-space-6); | ||||||
|  |         margin-bottom: var(--ha-space-3); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       h3, | ||||||
|  |       h4 { | ||||||
|  |         margin-top: var(--ha-space-4); | ||||||
|  |         margin-bottom: var(--ha-space-2); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       p { | ||||||
|  |         margin: var(--ha-space-2) 0; | ||||||
|  |         line-height: 1.6; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       .subtitle { | ||||||
|  |         color: var(--secondary-text-color); | ||||||
|  |         font-size: 1.1em; | ||||||
|  |         margin-bottom: var(--ha-space-4); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       table { | ||||||
|  |         width: 100%; | ||||||
|  |         border-collapse: collapse; | ||||||
|  |         margin: var(--ha-space-3) 0; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       th, | ||||||
|  |       td { | ||||||
|  |         text-align: left; | ||||||
|  |         padding: var(--ha-space-2); | ||||||
|  |         border-bottom: 1px solid var(--divider-color); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       th { | ||||||
|  |         font-weight: 500; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       code { | ||||||
|  |         background-color: var(--secondary-background-color); | ||||||
|  |         padding: 2px 6px; | ||||||
|  |         border-radius: 4px; | ||||||
|  |         font-family: monospace; | ||||||
|  |         font-size: 0.9em; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       pre { | ||||||
|  |         background-color: var(--secondary-background-color); | ||||||
|  |         padding: var(--ha-space-3); | ||||||
|  |         border-radius: 8px; | ||||||
|  |         overflow-x: auto; | ||||||
|  |         margin: var(--ha-space-3) 0; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       pre code { | ||||||
|  |         background-color: transparent; | ||||||
|  |         padding: 0; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       .buttons { | ||||||
|  |         display: flex; | ||||||
|  |         flex-direction: row; | ||||||
|  |         flex-wrap: wrap; | ||||||
|  |         gap: var(--ha-space-2); | ||||||
|  |         margin: var(--ha-space-4) 0; | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       a { | ||||||
|  |         color: var(--primary-color); | ||||||
|  |       } | ||||||
|  |     `, | ||||||
|  |   ]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-components-ha-wa-dialog": DemoHaWaDialog; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -11,6 +11,7 @@ import { provideHass } from "../../../../src/fake_data/provide_hass"; | |||||||
| import "../../components/demo-cards"; | import "../../components/demo-cards"; | ||||||
| import { mockIcons } from "../../../../demo/src/stubs/icons"; | import { mockIcons } from "../../../../demo/src/stubs/icons"; | ||||||
| import { ClimateEntityFeature } from "../../../../src/data/climate"; | import { ClimateEntityFeature } from "../../../../src/data/climate"; | ||||||
|  | import { FanEntityFeature } from "../../../../src/data/fan"; | ||||||
|  |  | ||||||
| const ENTITIES = [ | const ENTITIES = [ | ||||||
|   getEntity("switch", "tv_outlet", "on", { |   getEntity("switch", "tv_outlet", "on", { | ||||||
| @@ -100,6 +101,15 @@ const ENTITIES = [ | |||||||
|       ClimateEntityFeature.FAN_MODE + |       ClimateEntityFeature.FAN_MODE + | ||||||
|       ClimateEntityFeature.TARGET_TEMPERATURE_RANGE, |       ClimateEntityFeature.TARGET_TEMPERATURE_RANGE, | ||||||
|   }), |   }), | ||||||
|  |   getEntity("fan", "fan_demo", "on", { | ||||||
|  |     friendly_name: "Ceiling fan", | ||||||
|  |     device_class: "fan", | ||||||
|  |     direction: "reverse", | ||||||
|  |     supported_features: | ||||||
|  |       FanEntityFeature.DIRECTION + | ||||||
|  |       FanEntityFeature.SET_SPEED + | ||||||
|  |       FanEntityFeature.OSCILLATE, | ||||||
|  |   }), | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| const CONFIGS = [ | const CONFIGS = [ | ||||||
| @@ -261,6 +271,33 @@ const CONFIGS = [ | |||||||
|   - type: target-temperature |   - type: target-temperature | ||||||
|     `, |     `, | ||||||
|   }, |   }, | ||||||
|  |   { | ||||||
|  |     heading: "Fan direction feature", | ||||||
|  |     config: ` | ||||||
|  | - type: tile | ||||||
|  |   entity: fan.fan_demo | ||||||
|  |   features: | ||||||
|  |   - type: fan-direction | ||||||
|  |     `, | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     heading: "Fan speed feature", | ||||||
|  |     config: ` | ||||||
|  | - type: tile | ||||||
|  |   entity: fan.fan_demo | ||||||
|  |   features: | ||||||
|  |   - type: fan-speed | ||||||
|  |     `, | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     heading: "Fan oscillate feature", | ||||||
|  |     config: ` | ||||||
|  | - type: tile | ||||||
|  |   entity: fan.fan_demo | ||||||
|  |   features: | ||||||
|  |   - type: fan-oscillate | ||||||
|  |     `, | ||||||
|  |   }, | ||||||
| ]; | ]; | ||||||
|  |  | ||||||
| @customElement("demo-lovelace-tile-card") | @customElement("demo-lovelace-tile-card") | ||||||
|   | |||||||
| @@ -5,13 +5,13 @@ import type { | |||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property } from "lit/decorators"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
|  | import { mockIcons } from "../../../../demo/src/stubs/icons"; | ||||||
| import { computeDomain } from "../../../../src/common/entity/compute_domain"; | import { computeDomain } from "../../../../src/common/entity/compute_domain"; | ||||||
| import { computeStateDisplay } from "../../../../src/common/entity/compute_state_display"; | import { computeStateDisplay } from "../../../../src/common/entity/compute_state_display"; | ||||||
| import "../../../../src/components/data-table/ha-data-table"; | import "../../../../src/components/data-table/ha-data-table"; | ||||||
| import type { DataTableColumnContainer } from "../../../../src/components/data-table/ha-data-table"; | import type { DataTableColumnContainer } from "../../../../src/components/data-table/ha-data-table"; | ||||||
| import "../../../../src/components/entity/state-badge"; | import "../../../../src/components/entity/state-badge"; | ||||||
| import { provideHass } from "../../../../src/fake_data/provide_hass"; | import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||||
| import { mockIcons } from "../../../../demo/src/stubs/icons"; |  | ||||||
| import type { HomeAssistant } from "../../../../src/types"; | import type { HomeAssistant } from "../../../../src/types"; | ||||||
|  |  | ||||||
| const SENSOR_DEVICE_CLASSES = [ | const SENSOR_DEVICE_CLASSES = [ | ||||||
| @@ -434,7 +434,7 @@ export class DemoEntityState extends LitElement { | |||||||
|       display: block; |       display: block; | ||||||
|       height: 20px; |       height: 20px; | ||||||
|       width: 20px; |       width: 20px; | ||||||
|       border-radius: 10px; |       border-radius: var(--ha-border-radius-md); | ||||||
|       background-color: rgb(--color); |       background-color: rgb(--color); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement } from "lit/decorators"; | import { customElement } from "lit/decorators"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
| import type { ActionHandlerEvent } from "../../../../src/data/lovelace/action_handler"; | import type { ActionHandlerEvent } from "../../../../src/data/lovelace/action_handler"; | ||||||
| import { actionHandler } from "../../../../src/panels/lovelace/common/directives/action-handler-directive"; | import { actionHandler } from "../../../../src/panels/lovelace/common/directives/action-handler-directive"; | ||||||
| @@ -13,12 +13,16 @@ export class DemoUtilLongPress extends LitElement { | |||||||
|       ${[1, 2, 3].map( |       ${[1, 2, 3].map( | ||||||
|         () => html` |         () => html` | ||||||
|           <ha-card> |           <ha-card> | ||||||
|             <mwc-button |             <ha-button | ||||||
|  |               appearance="plain" | ||||||
|               @action=${this._handleAction} |               @action=${this._handleAction} | ||||||
|               .actionHandler=${actionHandler({})} |               .actionHandler=${actionHandler({ | ||||||
|  |                 hasHold: true, | ||||||
|  |                 hasDoubleClick: true, | ||||||
|  |               })} | ||||||
|             > |             > | ||||||
|               (long) press me! |               (long) press me! | ||||||
|             </mwc-button> |             </ha-button> | ||||||
|  |  | ||||||
|             <textarea></textarea> |             <textarea></textarea> | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								gallery/src/pages/more-info/fan.markdown
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										3
									
								
								gallery/src/pages/more-info/fan.markdown
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | --- | ||||||
|  | title: Fan | ||||||
|  | --- | ||||||
							
								
								
									
										50
									
								
								gallery/src/pages/more-info/fan.ts
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										50
									
								
								gallery/src/pages/more-info/fan.ts
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,50 @@ | |||||||
|  | import type { PropertyValues, TemplateResult } from "lit"; | ||||||
|  | import { html, LitElement } from "lit"; | ||||||
|  | import { customElement, property, query } from "lit/decorators"; | ||||||
|  | import "../../../../src/components/ha-card"; | ||||||
|  | import "../../../../src/dialogs/more-info/more-info-content"; | ||||||
|  | import { getEntity } from "../../../../src/fake_data/entity"; | ||||||
|  | import type { MockHomeAssistant } from "../../../../src/fake_data/provide_hass"; | ||||||
|  | import { provideHass } from "../../../../src/fake_data/provide_hass"; | ||||||
|  | import "../../components/demo-more-infos"; | ||||||
|  | import { FanEntityFeature } from "../../../../src/data/fan"; | ||||||
|  |  | ||||||
|  | const ENTITIES = [ | ||||||
|  |   getEntity("fan", "fan", "on", { | ||||||
|  |     friendly_name: "Fan", | ||||||
|  |     device_class: "fan", | ||||||
|  |     supported_features: | ||||||
|  |       FanEntityFeature.OSCILLATE + | ||||||
|  |       FanEntityFeature.DIRECTION + | ||||||
|  |       FanEntityFeature.SET_SPEED, | ||||||
|  |   }), | ||||||
|  | ]; | ||||||
|  |  | ||||||
|  | @customElement("demo-more-info-fan") | ||||||
|  | class DemoMoreInfoFan extends LitElement { | ||||||
|  |   @property({ attribute: false }) 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); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HTMLElementTagNameMap { | ||||||
|  |     "demo-more-info-fan": DemoMoreInfoFan; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -11,7 +11,10 @@ import "../../../../src/components/ha-alert"; | |||||||
| import "../../../../src/components/ha-button-menu"; | import "../../../../src/components/ha-button-menu"; | ||||||
| import "../../../../src/components/ha-card"; | import "../../../../src/components/ha-card"; | ||||||
| import "../../../../src/components/ha-form/ha-form"; | import "../../../../src/components/ha-form/ha-form"; | ||||||
| import type { HaFormSchema } from "../../../../src/components/ha-form/types"; | import type { | ||||||
|  |   HaFormSchema, | ||||||
|  |   HaFormDataContainer, | ||||||
|  | } from "../../../../src/components/ha-form/types"; | ||||||
| import "../../../../src/components/ha-formfield"; | import "../../../../src/components/ha-formfield"; | ||||||
| import "../../../../src/components/ha-icon-button"; | import "../../../../src/components/ha-icon-button"; | ||||||
| import "../../../../src/components/ha-list-item"; | import "../../../../src/components/ha-list-item"; | ||||||
| @@ -33,6 +36,7 @@ import { haStyle } from "../../../../src/resources/styles"; | |||||||
| import type { HomeAssistant } from "../../../../src/types"; | 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"; | ||||||
|  | import type { ObjectSelector, Selector } from "../../../../src/data/selector"; | ||||||
|  |  | ||||||
| const SUPPORTED_UI_TYPES = [ | const SUPPORTED_UI_TYPES = [ | ||||||
|   "string", |   "string", | ||||||
| @@ -78,78 +82,125 @@ class HassioAddonConfig extends LitElement { | |||||||
|  |  | ||||||
|   @query("ha-yaml-editor") private _editor?: HaYamlEditor; |   @query("ha-yaml-editor") private _editor?: HaYamlEditor; | ||||||
|  |  | ||||||
|   public computeLabel = (entry: HaFormSchema): string => |   private _getTranslationEntry( | ||||||
|     this.addon.translations[this.hass.language]?.configuration?.[entry.name] |     language: string, | ||||||
|       ?.name || |     entry: HaFormSchema, | ||||||
|     this.addon.translations.en?.configuration?.[entry.name]?.name || |     options?: { path?: string[] } | ||||||
|  |   ) { | ||||||
|  |     let parent = this.addon.translations[language]?.configuration; | ||||||
|  |     if (!parent) return undefined; | ||||||
|  |     if (options?.path) { | ||||||
|  |       for (const key of options.path) { | ||||||
|  |         parent = parent[key]?.fields; | ||||||
|  |         if (!parent) return undefined; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |     return parent[entry.name]; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   public computeLabel = ( | ||||||
|  |     entry: HaFormSchema, | ||||||
|  |     _data: HaFormDataContainer, | ||||||
|  |     options?: { path?: string[] } | ||||||
|  |   ): string => | ||||||
|  |     this._getTranslationEntry(this.hass.language, entry, options)?.name || | ||||||
|  |     this._getTranslationEntry("en", entry, options)?.name || | ||||||
|     entry.name; |     entry.name; | ||||||
|  |  | ||||||
|   public computeHelper = (entry: HaFormSchema): string => |   public computeHelper = ( | ||||||
|     this.addon.translations[this.hass.language]?.configuration?.[entry.name] |     entry: HaFormSchema, | ||||||
|  |     options?: { path?: string[] } | ||||||
|  |   ): string => | ||||||
|  |     this._getTranslationEntry(this.hass.language, entry, options) | ||||||
|       ?.description || |       ?.description || | ||||||
|     this.addon.translations.en?.configuration?.[entry.name]?.description || |     this._getTranslationEntry("en", entry, options)?.description || | ||||||
|     ""; |     ""; | ||||||
|  |  | ||||||
|   private _convertSchema = memoizeOne( |   private _convertSchema = memoizeOne( | ||||||
|     // Convert supervisor schema to selectors |     // Convert supervisor schema to selectors | ||||||
|     (schema: Record<string, any>): HaFormSchema[] => |     (schema: readonly HaFormSchema[]): HaFormSchema[] => | ||||||
|       schema.map((entry) => |       this._convertSchemaElements(schema) | ||||||
|         entry.type === "select" |  | ||||||
|           ? { |  | ||||||
|               name: entry.name, |  | ||||||
|               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 |  | ||||||
|                         ? 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", |  | ||||||
|                           step: entry.type === "float" ? "any" : undefined, |  | ||||||
|                         }, |  | ||||||
|                       }, |  | ||||||
|                     } |  | ||||||
|                   : entry |  | ||||||
|       ) |  | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   private _filteredShchema = memoizeOne( |   private _convertSchemaElements( | ||||||
|  |     schema: readonly HaFormSchema[] | ||||||
|  |   ): HaFormSchema[] { | ||||||
|  |     return schema.map((entry) => this._convertSchemaElement(entry)); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _convertSchemaElement(entry: any): HaFormSchema { | ||||||
|  |     if (entry.type === "schema" && !entry.multiple) { | ||||||
|  |       return { | ||||||
|  |         name: entry.name, | ||||||
|  |         type: "expandable", | ||||||
|  |         required: entry.required, | ||||||
|  |         schema: this._convertSchemaElements(entry.schema), | ||||||
|  |       }; | ||||||
|  |     } | ||||||
|  |     const selector = this._convertSchemaElementToSelector(entry, false); | ||||||
|  |     if (selector) { | ||||||
|  |       return { | ||||||
|  |         name: entry.name, | ||||||
|  |         required: entry.required, | ||||||
|  |         selector, | ||||||
|  |       }; | ||||||
|  |     } | ||||||
|  |     return entry; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _convertSchemaElementToSelector( | ||||||
|  |     entry: any, | ||||||
|  |     force: boolean | ||||||
|  |   ): Selector | null { | ||||||
|  |     if (entry.type === "select") { | ||||||
|  |       return { select: { options: entry.options } }; | ||||||
|  |     } | ||||||
|  |     if (entry.type === "string") { | ||||||
|  |       return entry.multiple | ||||||
|  |         ? { select: { options: [], multiple: true, custom_value: true } } | ||||||
|  |         : { | ||||||
|  |             text: { | ||||||
|  |               type: entry.format | ||||||
|  |                 ? entry.format | ||||||
|  |                 : MASKED_FIELDS.includes(entry.name) | ||||||
|  |                   ? "password" | ||||||
|  |                   : "text", | ||||||
|  |             }, | ||||||
|  |           }; | ||||||
|  |     } | ||||||
|  |     if (entry.type === "boolean") { | ||||||
|  |       return { boolean: {} }; | ||||||
|  |     } | ||||||
|  |     if (entry.type === "schema") { | ||||||
|  |       const fields: NonNullable<ObjectSelector["object"]>["fields"] = {}; | ||||||
|  |       for (const child_entry of entry.schema) { | ||||||
|  |         fields[child_entry.name] = { | ||||||
|  |           required: child_entry.required, | ||||||
|  |           selector: this._convertSchemaElementToSelector(child_entry, true)!, | ||||||
|  |         }; | ||||||
|  |       } | ||||||
|  |       return { | ||||||
|  |         object: { | ||||||
|  |           multiple: entry.multiple, | ||||||
|  |           fields, | ||||||
|  |         }, | ||||||
|  |       }; | ||||||
|  |     } | ||||||
|  |     if (entry.type === "float" || entry.type === "integer") { | ||||||
|  |       return { | ||||||
|  |         number: { | ||||||
|  |           mode: "box", | ||||||
|  |           step: entry.type === "float" ? "any" : undefined, | ||||||
|  |         }, | ||||||
|  |       }; | ||||||
|  |     } | ||||||
|  |     if (force) { | ||||||
|  |       return { object: {} }; | ||||||
|  |     } | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _filteredSchema = memoizeOne( | ||||||
|     (options: Record<string, unknown>, schema: HaFormSchema[]) => |     (options: Record<string, unknown>, schema: HaFormSchema[]) => | ||||||
|       schema.filter((entry) => entry.name in options || entry.required) |       schema.filter((entry) => entry.name in options || entry.required) | ||||||
|   ); |   ); | ||||||
| @@ -161,7 +212,7 @@ class HassioAddonConfig extends LitElement { | |||||||
|       showForm && |       showForm && | ||||||
|       JSON.stringify(this.addon.schema) !== |       JSON.stringify(this.addon.schema) !== | ||||||
|         JSON.stringify( |         JSON.stringify( | ||||||
|           this._filteredShchema(this.addon.options, this.addon.schema!) |           this._filteredSchema(this.addon.options, this.addon.schema!) | ||||||
|         ); |         ); | ||||||
|     return html` |     return html` | ||||||
|       <h1>${this.addon.name}</h1> |       <h1>${this.addon.name}</h1> | ||||||
| @@ -199,6 +250,7 @@ class HassioAddonConfig extends LitElement { | |||||||
|         <div class="card-content"> |         <div class="card-content"> | ||||||
|           ${showForm |           ${showForm | ||||||
|             ? html`<ha-form |             ? html`<ha-form | ||||||
|  |                 .hass=${this.hass} | ||||||
|                 .disabled=${this.disabled} |                 .disabled=${this.disabled} | ||||||
|                 .data=${this._options!} |                 .data=${this._options!} | ||||||
|                 @value-changed=${this._configChanged} |                 @value-changed=${this._configChanged} | ||||||
| @@ -207,7 +259,7 @@ class HassioAddonConfig extends LitElement { | |||||||
|                 .schema=${this._convertSchema( |                 .schema=${this._convertSchema( | ||||||
|                   this._showOptional |                   this._showOptional | ||||||
|                     ? this.addon.schema! |                     ? this.addon.schema! | ||||||
|                     : this._filteredShchema( |                     : this._filteredSchema( | ||||||
|                         this.addon.options, |                         this.addon.options, | ||||||
|                         this.addon.schema! |                         this.addon.schema! | ||||||
|                       ) |                       ) | ||||||
|   | |||||||
| @@ -99,7 +99,8 @@ class HassioAddonNetwork extends LitElement { | |||||||
|           : nothing} |           : nothing} | ||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           <ha-progress-button |           <ha-progress-button | ||||||
|             class="warning" |             variant="danger" | ||||||
|  |             appearance="plain" | ||||||
|             .disabled=${this.disabled} |             .disabled=${this.disabled} | ||||||
|             @click=${this._resetTapped} |             @click=${this._resetTapped} | ||||||
|           > |           > | ||||||
|   | |||||||
| @@ -25,6 +25,7 @@ import type { CSSResultGroup, TemplateResult } from "lit"; | |||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import { classMap } from "lit/directives/class-map"; | import { classMap } from "lit/directives/class-map"; | ||||||
|  | import { ifDefined } from "lit/directives/if-defined"; | ||||||
| import memoizeOne from "memoize-one"; | 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"; | ||||||
| @@ -187,12 +188,13 @@ class HassioAddonInfo extends LitElement { | |||||||
|                 "addon.dashboard.protection_mode.content" |                 "addon.dashboard.protection_mode.content" | ||||||
|               )} |               )} | ||||||
|               <ha-button |               <ha-button | ||||||
|  |                 variant="danger" | ||||||
|                 slot="action" |                 slot="action" | ||||||
|                 .label=${this.supervisor.localize( |  | ||||||
|                   "addon.dashboard.protection_mode.enable" |  | ||||||
|                 )} |  | ||||||
|                 @click=${this._protectionToggled} |                 @click=${this._protectionToggled} | ||||||
|               > |               > | ||||||
|  |                 ${this.supervisor.localize( | ||||||
|  |                   "addon.dashboard.protection_mode.enable" | ||||||
|  |                 )} | ||||||
|               </ha-button> |               </ha-button> | ||||||
|             </ha-alert> |             </ha-alert> | ||||||
|           ` |           ` | ||||||
| @@ -692,14 +694,16 @@ class HassioAddonInfo extends LitElement { | |||||||
|               ? this._computeIsRunning |               ? this._computeIsRunning | ||||||
|                 ? html` |                 ? html` | ||||||
|                     <ha-progress-button |                     <ha-progress-button | ||||||
|                       class="warning" |                       variant="danger" | ||||||
|  |                       appearance="plain" | ||||||
|                       @click=${this._stopClicked} |                       @click=${this._stopClicked} | ||||||
|                       .disabled=${systemManaged && !this.controlEnabled} |                       .disabled=${systemManaged && !this.controlEnabled} | ||||||
|                     > |                     > | ||||||
|                       ${this.supervisor.localize("addon.dashboard.stop")} |                       ${this.supervisor.localize("addon.dashboard.stop")} | ||||||
|                     </ha-progress-button> |                     </ha-progress-button> | ||||||
|                     <ha-progress-button |                     <ha-progress-button | ||||||
|                       class="warning" |                       variant="danger" | ||||||
|  |                       appearance="plain" | ||||||
|                       @click=${this._restartClicked} |                       @click=${this._restartClicked} | ||||||
|                     > |                     > | ||||||
|                       ${this.supervisor.localize("addon.dashboard.restart")} |                       ${this.supervisor.localize("addon.dashboard.restart")} | ||||||
| @@ -709,10 +713,60 @@ class HassioAddonInfo extends LitElement { | |||||||
|                     <ha-progress-button |                     <ha-progress-button | ||||||
|                       @click=${this._startClicked} |                       @click=${this._startClicked} | ||||||
|                       .progress=${this.addon.state === "startup"} |                       .progress=${this.addon.state === "startup"} | ||||||
|  |                       appearance="plain" | ||||||
|                     > |                     > | ||||||
|                       ${this.supervisor.localize("addon.dashboard.start")} |                       ${this.supervisor.localize("addon.dashboard.start")} | ||||||
|                     </ha-progress-button> |                     </ha-progress-button> | ||||||
|                   ` |                   ` | ||||||
|  |               : nothing} | ||||||
|  |           </div> | ||||||
|  |           <div> | ||||||
|  |             ${this.addon.version | ||||||
|  |               ? html` | ||||||
|  |                   <ha-progress-button | ||||||
|  |                     variant="danger" | ||||||
|  |                     appearance="plain" | ||||||
|  |                     @click=${this._uninstallClicked} | ||||||
|  |                     .disabled=${systemManaged && !this.controlEnabled} | ||||||
|  |                   > | ||||||
|  |                     ${this.supervisor.localize("addon.dashboard.uninstall")} | ||||||
|  |                   </ha-progress-button> | ||||||
|  |                   ${this.addon.build | ||||||
|  |                     ? html` | ||||||
|  |                         <ha-progress-button | ||||||
|  |                           variant="danger" | ||||||
|  |                           appearance="plain" | ||||||
|  |                           @click=${this._rebuildClicked} | ||||||
|  |                         > | ||||||
|  |                           ${this.supervisor.localize("addon.dashboard.rebuild")} | ||||||
|  |                         </ha-progress-button> | ||||||
|  |                       ` | ||||||
|  |                     : nothing} | ||||||
|  |                   ${this._computeShowWebUI || this._computeShowIngressUI | ||||||
|  |                     ? html` | ||||||
|  |                         <ha-button | ||||||
|  |                           href=${ifDefined( | ||||||
|  |                             !this._computeShowIngressUI | ||||||
|  |                               ? this._pathWebui! | ||||||
|  |                               : nothing | ||||||
|  |                           )} | ||||||
|  |                           target=${ifDefined( | ||||||
|  |                             !this._computeShowIngressUI ? "_blank" : nothing | ||||||
|  |                           )} | ||||||
|  |                           rel=${ifDefined( | ||||||
|  |                             !this._computeShowIngressUI ? "noopener" : nothing | ||||||
|  |                           )} | ||||||
|  |                           @click=${!this._computeShowWebUI | ||||||
|  |                             ? this._openIngress | ||||||
|  |                             : undefined} | ||||||
|  |                         > | ||||||
|  |                           ${this.supervisor.localize( | ||||||
|  |                             "addon.dashboard.open_web_ui" | ||||||
|  |                           )} | ||||||
|  |                         </ha-button> | ||||||
|  |                       ` | ||||||
|  |                     : nothing} | ||||||
|  |                 ` | ||||||
|               : html` |               : html` | ||||||
|                   <ha-progress-button |                   <ha-progress-button | ||||||
|                     .disabled=${!this.addon.available} |                     .disabled=${!this.addon.available} | ||||||
| @@ -722,58 +776,12 @@ class HassioAddonInfo extends LitElement { | |||||||
|                   </ha-progress-button> |                   </ha-progress-button> | ||||||
|                 `} |                 `} | ||||||
|           </div> |           </div> | ||||||
|           <div> |  | ||||||
|             ${this.addon.version |  | ||||||
|               ? html` ${this._computeShowWebUI |  | ||||||
|                     ? html` |  | ||||||
|                         <a |  | ||||||
|                           href=${this._pathWebui!} |  | ||||||
|                           tabindex="-1" |  | ||||||
|                           target="_blank" |  | ||||||
|                           rel="noopener" |  | ||||||
|                         > |  | ||||||
|                           <ha-button> |  | ||||||
|                             ${this.supervisor.localize( |  | ||||||
|                               "addon.dashboard.open_web_ui" |  | ||||||
|                             )} |  | ||||||
|                           </ha-button> |  | ||||||
|                         </a> |  | ||||||
|                       ` |  | ||||||
|                     : nothing} |  | ||||||
|                   ${this._computeShowIngressUI |  | ||||||
|                     ? html` |  | ||||||
|                         <ha-button @click=${this._openIngress}> |  | ||||||
|                           ${this.supervisor.localize( |  | ||||||
|                             "addon.dashboard.open_web_ui" |  | ||||||
|                           )} |  | ||||||
|                         </ha-button> |  | ||||||
|                       ` |  | ||||||
|                     : nothing} |  | ||||||
|                   <ha-progress-button |  | ||||||
|                     class="warning" |  | ||||||
|                     @click=${this._uninstallClicked} |  | ||||||
|                     .disabled=${systemManaged && !this.controlEnabled} |  | ||||||
|                   > |  | ||||||
|                     ${this.supervisor.localize("addon.dashboard.uninstall")} |  | ||||||
|                   </ha-progress-button> |  | ||||||
|                   ${this.addon.build |  | ||||||
|                     ? html` |  | ||||||
|                         <ha-progress-button |  | ||||||
|                           class="warning" |  | ||||||
|                           @click=${this._rebuildClicked} |  | ||||||
|                         > |  | ||||||
|                           ${this.supervisor.localize("addon.dashboard.rebuild")} |  | ||||||
|                         </ha-progress-button> |  | ||||||
|                       ` |  | ||||||
|                     : nothing}` |  | ||||||
|               : nothing} |  | ||||||
|           </div> |  | ||||||
|         </div> |         </div> | ||||||
|       </ha-card> |       </ha-card> | ||||||
|  |  | ||||||
|       ${this.addon.long_description |       ${this.addon.long_description | ||||||
|         ? html` |         ? html` | ||||||
|             <ha-card outlined> |             <ha-card class="long-description" outlined> | ||||||
|               <div class="card-content"> |               <div class="card-content"> | ||||||
|                 <ha-markdown |                 <ha-markdown | ||||||
|                   .content=${this.addon.long_description} |                   .content=${this.addon.long_description} | ||||||
| @@ -1146,15 +1154,17 @@ class HassioAddonInfo extends LitElement { | |||||||
|           ), |           ), | ||||||
|           dismissText: this.supervisor.localize("common.cancel"), |           dismissText: this.supervisor.localize("common.cancel"), | ||||||
|         }); |         }); | ||||||
|  |         button.actionError(); | ||||||
|         button.progress = false; |         button.progress = false; | ||||||
|         return; |         return; | ||||||
|       } |       } | ||||||
|     } catch (err: any) { |     } catch (err: any) { | ||||||
|  |       button.actionError(); | ||||||
|  |       button.progress = false; | ||||||
|       showAlertDialog(this, { |       showAlertDialog(this, { | ||||||
|         title: "Failed to validate addon configuration", |         title: "Failed to validate addon configuration", | ||||||
|         text: extractApiErrorMessage(err), |         text: extractApiErrorMessage(err), | ||||||
|       }); |       }); | ||||||
|       button.progress = false; |  | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -1168,11 +1178,15 @@ class HassioAddonInfo extends LitElement { | |||||||
|       }; |       }; | ||||||
|       fireEvent(this, "hass-api-called", eventdata); |       fireEvent(this, "hass-api-called", eventdata); | ||||||
|     } catch (err: any) { |     } catch (err: any) { | ||||||
|  |       button.actionError(); | ||||||
|  |       button.progress = false; | ||||||
|       showAlertDialog(this, { |       showAlertDialog(this, { | ||||||
|         title: this.supervisor.localize("addon.dashboard.action_error.start"), |         title: this.supervisor.localize("addon.dashboard.action_error.start"), | ||||||
|         text: extractApiErrorMessage(err), |         text: extractApiErrorMessage(err), | ||||||
|       }); |       }); | ||||||
|  |       return; | ||||||
|     } |     } | ||||||
|  |     button.actionSuccess(); | ||||||
|     button.progress = false; |     button.progress = false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -1228,6 +1242,7 @@ class HassioAddonInfo extends LitElement { | |||||||
|         path: "uninstall", |         path: "uninstall", | ||||||
|       }; |       }; | ||||||
|       fireEvent(this, "hass-api-called", eventdata); |       fireEvent(this, "hass-api-called", eventdata); | ||||||
|  |       button.actionSuccess(); | ||||||
|     } catch (err: any) { |     } catch (err: any) { | ||||||
|       showAlertDialog(this, { |       showAlertDialog(this, { | ||||||
|         title: this.supervisor.localize( |         title: this.supervisor.localize( | ||||||
| @@ -1235,6 +1250,7 @@ class HassioAddonInfo extends LitElement { | |||||||
|         ), |         ), | ||||||
|         text: extractApiErrorMessage(err), |         text: extractApiErrorMessage(err), | ||||||
|       }); |       }); | ||||||
|  |       button.actionError(); | ||||||
|     } |     } | ||||||
|     button.progress = false; |     button.progress = false; | ||||||
|   } |   } | ||||||
| @@ -1317,6 +1333,9 @@ class HassioAddonInfo extends LitElement { | |||||||
|         .description a { |         .description a { | ||||||
|           color: var(--primary-color); |           color: var(--primary-color); | ||||||
|         } |         } | ||||||
|  |         .long-description { | ||||||
|  |           direction: ltr; | ||||||
|  |         } | ||||||
|         ha-assist-chip { |         ha-assist-chip { | ||||||
|           --md-sys-color-primary: var(--text-primary-color); |           --md-sys-color-primary: var(--text-primary-color); | ||||||
|           --md-sys-color-on-surface: var(--text-primary-color); |           --md-sys-color-on-surface: var(--text-primary-color); | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
| import type { TemplateResult } from "lit"; | import type { TemplateResult } from "lit"; | ||||||
| import { LitElement, css, html, nothing } from "lit"; | import { LitElement, css, html, nothing } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property } from "lit/decorators"; | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
| import type { ActionDetail } from "@material/mwc-list"; | import type { ActionDetail } from "@material/mwc-list"; | ||||||
|  |  | ||||||
| import { mdiBackupRestore, mdiDelete, mdiDotsVertical, mdiPlus } from "@mdi/js"; | import { mdiBackupRestore, mdiDelete, mdiDotsVertical, mdiPlus } from "@mdi/js"; | ||||||
| @@ -17,6 +16,7 @@ import type { | |||||||
| } from "../../../src/components/data-table/ha-data-table"; | } from "../../../src/components/data-table/ha-data-table"; | ||||||
| import "../../../src/components/ha-button-menu"; | import "../../../src/components/ha-button-menu"; | ||||||
| import "../../../src/components/ha-fab"; | import "../../../src/components/ha-fab"; | ||||||
|  | import "../../../src/components/ha-button"; | ||||||
| import "../../../src/components/ha-icon-button"; | import "../../../src/components/ha-icon-button"; | ||||||
| import "../../../src/components/ha-list-item"; | import "../../../src/components/ha-list-item"; | ||||||
| import "../../../src/components/ha-svg-icon"; | import "../../../src/components/ha-svg-icon"; | ||||||
| @@ -241,12 +241,13 @@ export class HassioBackups extends LitElement { | |||||||
|               <div class="header-btns"> |               <div class="header-btns"> | ||||||
|                 ${!this.narrow |                 ${!this.narrow | ||||||
|                   ? html` |                   ? html` | ||||||
|                       <mwc-button |                       <ha-button | ||||||
|  |                         appearance="plain" | ||||||
|  |                         variant="danger" | ||||||
|                         @click=${this._deleteSelected} |                         @click=${this._deleteSelected} | ||||||
|                         class="warning" |  | ||||||
|                       > |                       > | ||||||
|                         ${this.supervisor.localize("backup.delete_selected")} |                         ${this.supervisor.localize("backup.delete_selected")} | ||||||
|                       </mwc-button> |                       </ha-button> | ||||||
|                     ` |                     ` | ||||||
|                   : html` |                   : html` | ||||||
|                       <ha-icon-button |                       <ha-icon-button | ||||||
| @@ -408,7 +409,7 @@ export class HassioBackups extends LitElement { | |||||||
|           margin-inline-end: -12px; |           margin-inline-end: -12px; | ||||||
|           margin-inline-start: initial; |           margin-inline-start: initial; | ||||||
|         } |         } | ||||||
|         .header-btns > mwc-button, |         .header-btns > ha-button, | ||||||
|         .header-btns > ha-icon-button { |         .header-btns > ha-icon-button { | ||||||
|           margin: 8px; |           margin: 8px; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -121,7 +121,7 @@ class HassioCardContent extends LitElement { | |||||||
|       height: 12px; |       height: 12px; | ||||||
|       top: 8px; |       top: 8px; | ||||||
|       right: 8px; |       right: 8px; | ||||||
|       border-radius: 50%; |       border-radius: var(--ha-border-radius-circle); | ||||||
|     } |     } | ||||||
|     .topbar { |     .topbar { | ||||||
|       position: absolute; |       position: absolute; | ||||||
|   | |||||||
| @@ -1,10 +1,9 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property } from "lit/decorators"; | ||||||
| import memoizeOne from "memoize-one"; | import memoizeOne from "memoize-one"; | ||||||
| import "../../../src/components/buttons/ha-progress-button"; |  | ||||||
| import "../../../src/components/ha-card"; | import "../../../src/components/ha-card"; | ||||||
|  | import "../../../src/components/ha-button"; | ||||||
| import "../../../src/components/ha-settings-row"; | import "../../../src/components/ha-settings-row"; | ||||||
| import "../../../src/components/ha-svg-icon"; | import "../../../src/components/ha-svg-icon"; | ||||||
| import type { HassioHassOSInfo } from "../../../src/data/hassio/host"; | import type { HassioHassOSInfo } from "../../../src/data/hassio/host"; | ||||||
| @@ -109,10 +108,9 @@ export class HassioUpdate extends LitElement { | |||||||
|           </ha-settings-row> |           </ha-settings-row> | ||||||
|         </div> |         </div> | ||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           <a href="/hassio/update-available/${key}"> |           <ha-button appearance="plain" href="/hassio/update-available/${key}"> | ||||||
|             <mwc-button .label=${this.supervisor.localize("common.show")}> |             ${this.supervisor.localize("common.show")} | ||||||
|             </mwc-button> |           </ha-button> | ||||||
|           </a> |  | ||||||
|         </div> |         </div> | ||||||
|       </ha-card> |       </ha-card> | ||||||
|     `; |     `; | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| import "@material/mwc-button/mwc-button"; |  | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } 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-dialog"; | import "../../../../src/components/ha-dialog"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-form/ha-form"; | import "../../../../src/components/ha-form/ha-form"; | ||||||
| import type { SchemaUnion } from "../../../../src/components/ha-form/types"; | import type { SchemaUnion } from "../../../../src/components/ha-form/types"; | ||||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||||
| @@ -77,20 +77,21 @@ class HassioBackupLocationDialog extends LitElement { | |||||||
|           @value-changed=${this._valueChanged} |           @value-changed=${this._valueChanged} | ||||||
|           dialogInitialFocus |           dialogInitialFocus | ||||||
|         ></ha-form> |         ></ha-form> | ||||||
|         <mwc-button |         <ha-button | ||||||
|  |           appearance="plain" | ||||||
|           slot="secondaryAction" |           slot="secondaryAction" | ||||||
|           @click=${this.closeDialog} |           @click=${this.closeDialog} | ||||||
|           dialogInitialFocus |           dialogInitialFocus | ||||||
|         > |         > | ||||||
|           ${this._dialogParams.supervisor.localize("common.cancel")} |           ${this._dialogParams.supervisor.localize("common.cancel")} | ||||||
|         </mwc-button> |         </ha-button> | ||||||
|         <mwc-button |         <ha-button | ||||||
|           .disabled=${this._waiting || !this._data} |           .disabled=${this._waiting || !this._data} | ||||||
|           slot="primaryAction" |           slot="primaryAction" | ||||||
|           @click=${this._changeMount} |           @click=${this._changeMount} | ||||||
|         > |         > | ||||||
|           ${this._dialogParams.supervisor.localize("common.save")} |           ${this._dialogParams.supervisor.localize("common.save")} | ||||||
|         </mwc-button> |         </ha-button> | ||||||
|       </ha-dialog> |       </ha-dialog> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -8,7 +8,6 @@ import { atLeastVersion } from "../../../../src/common/config/version"; | |||||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||||
| import { stopPropagation } from "../../../../src/common/dom/stop_propagation"; | import { stopPropagation } from "../../../../src/common/dom/stop_propagation"; | ||||||
| import { slugify } from "../../../../src/common/string/slugify"; | import { slugify } from "../../../../src/common/string/slugify"; | ||||||
| import "../../../../src/components/buttons/ha-progress-button"; |  | ||||||
| import "../../../../src/components/ha-alert"; | import "../../../../src/components/ha-alert"; | ||||||
| import "../../../../src/components/ha-button"; | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-button-menu"; | import "../../../../src/components/ha-button-menu"; | ||||||
|   | |||||||
| @@ -1,10 +1,9 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| 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/ha-alert"; | import "../../../../src/components/ha-alert"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-spinner"; | import "../../../../src/components/ha-spinner"; | ||||||
| import { createCloseHeading } from "../../../../src/components/ha-dialog"; | import { createCloseHeading } from "../../../../src/components/ha-dialog"; | ||||||
| import { | import { | ||||||
| @@ -69,16 +68,20 @@ class HassioCreateBackupDialog extends LitElement { | |||||||
|         ${this._error |         ${this._error | ||||||
|           ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` |           ? html`<ha-alert alert-type="error">${this._error}</ha-alert>` | ||||||
|           : ""} |           : ""} | ||||||
|         <mwc-button slot="secondaryAction" @click=${this.closeDialog}> |         <ha-button | ||||||
|  |           appearance="plain" | ||||||
|  |           slot="secondaryAction" | ||||||
|  |           @click=${this.closeDialog} | ||||||
|  |         > | ||||||
|           ${this._dialogParams.supervisor.localize("common.close")} |           ${this._dialogParams.supervisor.localize("common.close")} | ||||||
|         </mwc-button> |         </ha-button> | ||||||
|         <mwc-button |         <ha-button | ||||||
|           .disabled=${this._creatingBackup} |           .disabled=${this._creatingBackup} | ||||||
|           slot="primaryAction" |           slot="primaryAction" | ||||||
|           @click=${this._createBackup} |           @click=${this._createBackup} | ||||||
|         > |         > | ||||||
|           ${this._dialogParams.supervisor.localize("backup.create")} |           ${this._dialogParams.supervisor.localize("backup.create")} | ||||||
|         </mwc-button> |         </ha-button> | ||||||
|       </ha-dialog> |       </ha-dialog> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ 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-dialog"; | import "../../../../src/components/ha-dialog"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-list-item"; | import "../../../../src/components/ha-list-item"; | ||||||
| import "../../../../src/components/ha-select"; | import "../../../../src/components/ha-select"; | ||||||
| import "../../../../src/components/ha-spinner"; | import "../../../../src/components/ha-spinner"; | ||||||
| @@ -20,8 +21,8 @@ import type { HomeAssistant } from "../../../../src/types"; | |||||||
| import type { HassioDatatiskDialogParams } from "./show-dialog-hassio-datadisk"; | import type { HassioDatatiskDialogParams } from "./show-dialog-hassio-datadisk"; | ||||||
|  |  | ||||||
| const calculateMoveTime = memoizeOne((supervisor: Supervisor): number => { | const calculateMoveTime = memoizeOne((supervisor: Supervisor): number => { | ||||||
|   const speed = supervisor.host.disk_life_time !== "" ? 30 : 10; |   // Assume a speed of 30 MB/s. | ||||||
|   const moveTime = (supervisor.host.disk_used * 1000) / 60 / speed; |   const moveTime = (supervisor.host.disk_used * 1000) / 60 / 30; | ||||||
|   const rebootTime = (supervisor.host.startup_time * 4) / 60; |   const rebootTime = (supervisor.host.startup_time * 4) / 60; | ||||||
|   return Math.ceil((moveTime + rebootTime) / 10) * 10; |   return Math.ceil((moveTime + rebootTime) / 10) * 10; | ||||||
| }); | }); | ||||||
| @@ -109,17 +110,18 @@ class HassioDatadiskDialog extends LitElement { | |||||||
|                       "dialog.datadisk_move.no_devices" |                       "dialog.datadisk_move.no_devices" | ||||||
|                     )} |                     )} | ||||||
|  |  | ||||||
|               <mwc-button |               <ha-button | ||||||
|                 slot="secondaryAction" |                 appearance="plain" | ||||||
|  |                 slot="primaryAction" | ||||||
|                 @click=${this.closeDialog} |                 @click=${this.closeDialog} | ||||||
|                 dialogInitialFocus |                 dialogInitialFocus | ||||||
|               > |               > | ||||||
|                 ${this.dialogParams.supervisor.localize( |                 ${this.dialogParams.supervisor.localize( | ||||||
|                   "dialog.datadisk_move.cancel" |                   "dialog.datadisk_move.cancel" | ||||||
|                 )} |                 )} | ||||||
|               </mwc-button> |               </ha-button> | ||||||
|  |  | ||||||
|               <mwc-button |               <ha-button | ||||||
|                 .disabled=${!this.selectedDevice} |                 .disabled=${!this.selectedDevice} | ||||||
|                 slot="primaryAction" |                 slot="primaryAction" | ||||||
|                 @click=${this._moveDatadisk} |                 @click=${this._moveDatadisk} | ||||||
| @@ -127,7 +129,7 @@ class HassioDatadiskDialog extends LitElement { | |||||||
|                 ${this.dialogParams.supervisor.localize( |                 ${this.dialogParams.supervisor.localize( | ||||||
|                   "dialog.datadisk_move.move" |                   "dialog.datadisk_move.move" | ||||||
|                 )} |                 )} | ||||||
|               </mwc-button>`} |               </ha-button>`} | ||||||
|       </ha-dialog> |       </ha-dialog> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -164,7 +164,7 @@ class HassioHardwareDialog extends LitElement { | |||||||
|         pre, |         pre, | ||||||
|         code { |         code { | ||||||
|           background-color: var(--markdown-code-background-color, none); |           background-color: var(--markdown-code-background-color, none); | ||||||
|           border-radius: 3px; |           border-radius: var(--ha-border-radius-sm); | ||||||
|         } |         } | ||||||
|         pre { |         pre { | ||||||
|           padding: 16px; |           padding: 16px; | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import "@material/mwc-button/mwc-button"; |  | ||||||
| import { mdiClose } from "@mdi/js"; | import { mdiClose } from "@mdi/js"; | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| @@ -6,6 +5,7 @@ import { customElement, property, state } from "lit/decorators"; | |||||||
| import { cache } from "lit/directives/cache"; | import { cache } from "lit/directives/cache"; | ||||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||||
| import "../../../../src/components/ha-alert"; | import "../../../../src/components/ha-alert"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-dialog"; | import "../../../../src/components/ha-dialog"; | ||||||
| import "../../../../src/components/ha-expansion-panel"; | import "../../../../src/components/ha-expansion-panel"; | ||||||
| import "../../../../src/components/ha-formfield"; | import "../../../../src/components/ha-formfield"; | ||||||
| @@ -15,7 +15,8 @@ import "../../../../src/components/ha-list"; | |||||||
| import "../../../../src/components/ha-list-item"; | import "../../../../src/components/ha-list-item"; | ||||||
| import "../../../../src/components/ha-password-field"; | import "../../../../src/components/ha-password-field"; | ||||||
| import "../../../../src/components/ha-radio"; | import "../../../../src/components/ha-radio"; | ||||||
| import "../../../../src/components/ha-spinner"; | import "../../../../src/components/ha-tab-group"; | ||||||
|  | import "../../../../src/components/ha-tab-group-tab"; | ||||||
| import "../../../../src/components/ha-textfield"; | import "../../../../src/components/ha-textfield"; | ||||||
| import type { HaTextField } from "../../../../src/components/ha-textfield"; | import type { HaTextField } from "../../../../src/components/ha-textfield"; | ||||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||||
| @@ -37,7 +38,6 @@ import type { HassDialog } from "../../../../src/dialogs/make-dialog-manager"; | |||||||
| import { haStyleDialog } from "../../../../src/resources/styles"; | import { haStyleDialog } from "../../../../src/resources/styles"; | ||||||
| import type { HomeAssistant } from "../../../../src/types"; | import type { HomeAssistant } from "../../../../src/types"; | ||||||
| import type { HassioNetworkDialogParams } from "./show-dialog-network"; | import type { HassioNetworkDialogParams } from "./show-dialog-network"; | ||||||
| import "../../../../src/components/sl-tab-group"; |  | ||||||
|  |  | ||||||
| const IP_VERSIONS = ["ipv4", "ipv6"]; | const IP_VERSIONS = ["ipv4", "ipv6"]; | ||||||
|  |  | ||||||
| @@ -115,19 +115,19 @@ export class DialogHassioNetwork | |||||||
|             ></ha-icon-button> |             ></ha-icon-button> | ||||||
|           </ha-header-bar> |           </ha-header-bar> | ||||||
|           ${this._interfaces.length > 1 |           ${this._interfaces.length > 1 | ||||||
|             ? html`<sl-tab-group @sl-tab-show=${this._handleTabActivated} |             ? html`<ha-tab-group @wa-tab-show=${this._handleTabActivated} | ||||||
|                 >${this._interfaces.map( |                 >${this._interfaces.map( | ||||||
|                   (device, index) => |                   (device, index) => | ||||||
|                     html`<sl-tab |                     html`<ha-tab-group-tab | ||||||
|                       slot="nav" |                       slot="nav" | ||||||
|                       .id=${device.interface} |                       .id=${device.interface} | ||||||
|                       .panel=${index.toString()} |                       .panel=${index.toString()} | ||||||
|                       .active=${this._curTabIndex === index} |                       .active=${this._curTabIndex === index} | ||||||
|                     > |                     > | ||||||
|                       ${device.interface} |                       ${device.interface} | ||||||
|                     </sl-tab>` |                     </ha-tab-group-tab>` | ||||||
|                 )} |                 )} | ||||||
|               </sl-tab-group>` |               </ha-tab-group>` | ||||||
|             : ""} |             : ""} | ||||||
|         </div> |         </div> | ||||||
|         ${cache(this._renderTab())} |         ${cache(this._renderTab())} | ||||||
| @@ -154,16 +154,16 @@ export class DialogHassioNetwork | |||||||
|                       )} |                       )} | ||||||
|                     </p>` |                     </p>` | ||||||
|                   : ""} |                   : ""} | ||||||
|                 <mwc-button |                 <ha-button | ||||||
|  |                   appearance="plain" | ||||||
|  |                   size="small" | ||||||
|                   class="scan" |                   class="scan" | ||||||
|                   @click=${this._scanForAP} |                   @click=${this._scanForAP} | ||||||
|                   .disabled=${this._scanning} |                   .disabled=${this._scanning} | ||||||
|  |                   .loading=${this._scanning} | ||||||
|                 > |                 > | ||||||
|                   ${this._scanning |                   ${this.supervisor.localize("dialog.network.scan_ap")} | ||||||
|                     ? html`<ha-spinner aria-label="Scanning" size="small"> |                 </ha-button> | ||||||
|                       </ha-spinner>` |  | ||||||
|                     : this.supervisor.localize("dialog.network.scan_ap")} |  | ||||||
|                 </mwc-button> |  | ||||||
|                 ${this._accessPoints && |                 ${this._accessPoints && | ||||||
|                 this._accessPoints.accesspoints && |                 this._accessPoints.accesspoints && | ||||||
|                 this._accessPoints.accesspoints.length !== 0 |                 this._accessPoints.accesspoints.length !== 0 | ||||||
| @@ -270,16 +270,16 @@ export class DialogHassioNetwork | |||||||
|           : ""} |           : ""} | ||||||
|       </div> |       </div> | ||||||
|       <div class="buttons"> |       <div class="buttons"> | ||||||
|         <mwc-button |         <ha-button @click=${this.closeDialog} appearance="plain"> | ||||||
|           .label=${this.supervisor.localize("common.cancel")} |           ${this.supervisor.localize("common.cancel")} | ||||||
|           @click=${this.closeDialog} |         </ha-button> | ||||||
|  |         <ha-button | ||||||
|  |           @click=${this._updateNetwork} | ||||||
|  |           .disabled=${!this._dirty} | ||||||
|  |           .loading=${this._processing} | ||||||
|         > |         > | ||||||
|         </mwc-button> |           ${this.supervisor.localize("common.save")} | ||||||
|         <mwc-button @click=${this._updateNetwork} .disabled=${!this._dirty}> |         </ha-button> | ||||||
|           ${this._processing |  | ||||||
|             ? html`<ha-spinner size="small"> </ha-spinner>` |  | ||||||
|             : this.supervisor.localize("common.save")} |  | ||||||
|         </mwc-button> |  | ||||||
|       </div>`; |       </div>`; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -584,11 +584,7 @@ export class DialogHassioNetwork | |||||||
|           } |           } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         mwc-button.warning { |         ha-button.scan { | ||||||
|           --mdc-theme-primary: var(--error-color); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         mwc-button.scan { |  | ||||||
|           margin-left: 8px; |           margin-left: 8px; | ||||||
|           margin-inline-start: 8px; |           margin-inline-start: 8px; | ||||||
|           margin-inline-end: initial; |           margin-inline-end: initial; | ||||||
| @@ -609,8 +605,8 @@ export class DialogHassioNetwork | |||||||
|             var(--mdc-dialog-scroll-divider-color, rgba(0, 0, 0, 0.12)); |             var(--mdc-dialog-scroll-divider-color, rgba(0, 0, 0, 0.12)); | ||||||
|           display: flex; |           display: flex; | ||||||
|           justify-content: space-between; |           justify-content: space-between; | ||||||
|           padding: 8px; |           padding: 16px; | ||||||
|           padding-bottom: max(var(--safe-area-inset-bottom), 8px); |           padding-bottom: max(var(--safe-area-inset-bottom), 16px); | ||||||
|           background-color: var(--mdc-theme-surface, #fff); |           background-color: var(--mdc-theme-surface, #fff); | ||||||
|         } |         } | ||||||
|         .warning { |         .warning { | ||||||
| @@ -632,10 +628,10 @@ export class DialogHassioNetwork | |||||||
|           --mdc-list-side-padding: 10px; |           --mdc-list-side-padding: 10px; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         sl-tab { |         ha-tab-group-tab { | ||||||
|           flex: 1; |           flex: 1; | ||||||
|         } |         } | ||||||
|         sl-tab::part(base) { |         ha-tab-group-tab::part(base) { | ||||||
|           width: 100%; |           width: 100%; | ||||||
|           justify-content: center; |           justify-content: center; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -1,13 +1,14 @@ | |||||||
| import "@material/mwc-button/mwc-button"; | import { mdiDelete, mdiPlus } from "@mdi/js"; | ||||||
| import { mdiDelete } from "@mdi/js"; |  | ||||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | import type { CSSResultGroup, TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
|  | import "../../../../src/components/ha-button"; | ||||||
| import { createCloseHeading } from "../../../../src/components/ha-dialog"; | import { createCloseHeading } from "../../../../src/components/ha-dialog"; | ||||||
| import "../../../../src/components/ha-form/ha-form"; | import "../../../../src/components/ha-form/ha-form"; | ||||||
| import type { SchemaUnion } from "../../../../src/components/ha-form/types"; | import type { SchemaUnion } from "../../../../src/components/ha-form/types"; | ||||||
| import "../../../../src/components/ha-icon-button"; | import "../../../../src/components/ha-icon-button"; | ||||||
| import "../../../../src/components/ha-settings-row"; | import "../../../../src/components/ha-settings-row"; | ||||||
|  | import "../../../../src/components/ha-svg-icon"; | ||||||
| import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | import { extractApiErrorMessage } from "../../../../src/data/hassio/common"; | ||||||
| import { | import { | ||||||
|   addHassioDockerRegistry, |   addHassioDockerRegistry, | ||||||
| @@ -84,16 +85,19 @@ class HassioRegistriesDialog extends LitElement { | |||||||
|                 dialogInitialFocus |                 dialogInitialFocus | ||||||
|               ></ha-form> |               ></ha-form> | ||||||
|               <div class="action"> |               <div class="action"> | ||||||
|                 <mwc-button |                 <ha-button | ||||||
|                   ?disabled=${Boolean( |                   ?disabled=${Boolean( | ||||||
|                     !this._input.registry || |                     !this._input.registry || | ||||||
|                       !this._input.username || |                       !this._input.username || | ||||||
|                       !this._input.password |                       !this._input.password | ||||||
|                   )} |                   )} | ||||||
|                   @click=${this._addNewRegistry} |                   @click=${this._addNewRegistry} | ||||||
|  |                   appearance="filled" | ||||||
|  |                   size="small" | ||||||
|                 > |                 > | ||||||
|  |                   <ha-svg-icon slot="start" .path=${mdiPlus}></ha-svg-icon> | ||||||
|                   ${this.supervisor.localize("dialog.registries.add_registry")} |                   ${this.supervisor.localize("dialog.registries.add_registry")} | ||||||
|                 </mwc-button> |                 </ha-button> | ||||||
|               </div> |               </div> | ||||||
|             ` |             ` | ||||||
|           : html`${this._registries?.length |           : html`${this._registries?.length | ||||||
| @@ -126,11 +130,17 @@ class HassioRegistriesDialog extends LitElement { | |||||||
|                     </ha-alert> |                     </ha-alert> | ||||||
|                   `} |                   `} | ||||||
|               <div class="action"> |               <div class="action"> | ||||||
|                 <mwc-button @click=${this._addRegistry} dialogInitialFocus> |                 <ha-button | ||||||
|  |                   @click=${this._addRegistry} | ||||||
|  |                   dialogInitialFocus | ||||||
|  |                   appearance="filled" | ||||||
|  |                   size="small" | ||||||
|  |                 > | ||||||
|  |                   <ha-svg-icon slot="start" .path=${mdiPlus}></ha-svg-icon> | ||||||
|                   ${this.supervisor.localize( |                   ${this.supervisor.localize( | ||||||
|                     "dialog.registries.add_new_registry" |                     "dialog.registries.add_new_registry" | ||||||
|                   )} |                   )} | ||||||
|                 </mwc-button> |                 </ha-button> | ||||||
|               </div> `} |               </div> `} | ||||||
|       </ha-dialog> |       </ha-dialog> | ||||||
|     `; |     `; | ||||||
| @@ -218,7 +228,7 @@ class HassioRegistriesDialog extends LitElement { | |||||||
|       css` |       css` | ||||||
|         .registry { |         .registry { | ||||||
|           border: 1px solid var(--divider-color); |           border: 1px solid var(--divider-color); | ||||||
|           border-radius: 4px; |           border-radius: var(--ha-border-radius-sm); | ||||||
|           margin-top: 4px; |           margin-top: 4px; | ||||||
|         } |         } | ||||||
|         .action { |         .action { | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| import "@material/mwc-button/mwc-button"; | import { mdiDelete, mdiDeleteOff, mdiPlus } from "@mdi/js"; | ||||||
| import { mdiDelete, mdiDeleteOff } from "@mdi/js"; |  | ||||||
| import type { CSSResultGroup } from "lit"; | import type { CSSResultGroup } from "lit"; | ||||||
| import { css, html, LitElement, nothing } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
| @@ -7,10 +6,15 @@ import memoizeOne from "memoize-one"; | |||||||
| import { fireEvent } from "../../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../../src/common/dom/fire_event"; | ||||||
| import { caseInsensitiveStringCompare } from "../../../../src/common/string/compare"; | import { caseInsensitiveStringCompare } from "../../../../src/common/string/compare"; | ||||||
| import "../../../../src/components/ha-alert"; | import "../../../../src/components/ha-alert"; | ||||||
| import "../../../../src/components/ha-tooltip"; | import "../../../../src/components/ha-button"; | ||||||
| import "../../../../src/components/ha-spinner"; |  | ||||||
| import { createCloseHeading } from "../../../../src/components/ha-dialog"; | import { createCloseHeading } from "../../../../src/components/ha-dialog"; | ||||||
| import "../../../../src/components/ha-icon-button"; | import "../../../../src/components/ha-icon-button"; | ||||||
|  | import "../../../../src/components/ha-md-list"; | ||||||
|  | import "../../../../src/components/ha-md-list-item"; | ||||||
|  | import "../../../../src/components/ha-svg-icon"; | ||||||
|  | import "../../../../src/components/ha-textfield"; | ||||||
|  | import type { HaTextField } from "../../../../src/components/ha-textfield"; | ||||||
|  | import "../../../../src/components/ha-tooltip"; | ||||||
| import type { | import type { | ||||||
|   HassioAddonInfo, |   HassioAddonInfo, | ||||||
|   HassioAddonRepository, |   HassioAddonRepository, | ||||||
| @@ -24,10 +28,6 @@ import { | |||||||
| import { haStyle, haStyleDialog } from "../../../../src/resources/styles"; | import { haStyle, haStyleDialog } from "../../../../src/resources/styles"; | ||||||
| import type { HomeAssistant } from "../../../../src/types"; | import type { HomeAssistant } from "../../../../src/types"; | ||||||
| import type { HassioRepositoryDialogParams } from "./show-dialog-repositories"; | import type { HassioRepositoryDialogParams } from "./show-dialog-repositories"; | ||||||
| import type { HaTextField } from "../../../../src/components/ha-textfield"; |  | ||||||
| import "../../../../src/components/ha-textfield"; |  | ||||||
| import "../../../../src/components/ha-md-list"; |  | ||||||
| import "../../../../src/components/ha-md-list-item"; |  | ||||||
|  |  | ||||||
| @customElement("dialog-hassio-repositories") | @customElement("dialog-hassio-repositories") | ||||||
| class HassioRepositoriesDialog extends LitElement { | class HassioRepositoriesDialog extends LitElement { | ||||||
| @@ -119,26 +119,27 @@ class HassioRepositoriesDialog extends LitElement { | |||||||
|                         <div>${repo.url}</div> |                         <div>${repo.url}</div> | ||||||
|                       </div> |                       </div> | ||||||
|                       <ha-tooltip |                       <ha-tooltip | ||||||
|  |                         .for="icon-button-${repo.slug}" | ||||||
|                         class="delete" |                         class="delete" | ||||||
|                         slot="end" |                         slot="end" | ||||||
|                         .content=${this._dialogParams!.supervisor.localize( |                       > | ||||||
|  |                         ${this._dialogParams!.supervisor.localize( | ||||||
|                           usedRepositories.includes(repo.slug) |                           usedRepositories.includes(repo.slug) | ||||||
|                             ? "dialog.repositories.used" |                             ? "dialog.repositories.used" | ||||||
|                             : "dialog.repositories.remove" |                             : "dialog.repositories.remove" | ||||||
|                         )} |                         )} | ||||||
|                       > |  | ||||||
|                         <div> |  | ||||||
|                           <ha-icon-button |  | ||||||
|                             .disabled=${usedRepositories.includes(repo.slug)} |  | ||||||
|                             .slug=${repo.slug} |  | ||||||
|                             .path=${usedRepositories.includes(repo.slug) |  | ||||||
|                               ? mdiDeleteOff |  | ||||||
|                               : mdiDelete} |  | ||||||
|                             @click=${this._removeRepository} |  | ||||||
|                           > |  | ||||||
|                           </ha-icon-button> |  | ||||||
|                         </div> |  | ||||||
|                       </ha-tooltip> |                       </ha-tooltip> | ||||||
|  |                       <div .id="icon-button-${repo.slug}"> | ||||||
|  |                         <ha-icon-button | ||||||
|  |                           .disabled=${usedRepositories.includes(repo.slug)} | ||||||
|  |                           .slug=${repo.slug} | ||||||
|  |                           .path=${usedRepositories.includes(repo.slug) | ||||||
|  |                             ? mdiDeleteOff | ||||||
|  |                             : mdiDelete} | ||||||
|  |                           @click=${this._removeRepository} | ||||||
|  |                         > | ||||||
|  |                         </ha-icon-button> | ||||||
|  |                       </div> | ||||||
|                     </ha-md-list-item> |                     </ha-md-list-item> | ||||||
|                   ` |                   ` | ||||||
|                 ) |                 ) | ||||||
| @@ -159,18 +160,22 @@ class HassioRepositoriesDialog extends LitElement { | |||||||
|               @keydown=${this._handleKeyAdd} |               @keydown=${this._handleKeyAdd} | ||||||
|               dialogInitialFocus |               dialogInitialFocus | ||||||
|             ></ha-textfield> |             ></ha-textfield> | ||||||
|             <mwc-button @click=${this._addRepository}> |             <ha-button | ||||||
|               ${this._processing |               .loading=${this._processing} | ||||||
|                 ? html`<ha-spinner size="small"></ha-spinner>` |               @click=${this._addRepository} | ||||||
|                 : this._dialogParams!.supervisor.localize( |               appearance="filled" | ||||||
|                     "dialog.repositories.add" |               size="small" | ||||||
|                   )} |             > | ||||||
|             </mwc-button> |               <ha-svg-icon slot="start" .path=${mdiPlus}></ha-svg-icon> | ||||||
|  |               ${this._dialogParams!.supervisor.localize( | ||||||
|  |                 "dialog.repositories.add" | ||||||
|  |               )} | ||||||
|  |             </ha-button> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <mwc-button slot="primaryAction" @click=${this.closeDialog}> |         <ha-button slot="primaryAction" @click=${this.closeDialog}> | ||||||
|           ${this._dialogParams?.supervisor.localize("common.close")} |           ${this._dialogParams?.supervisor.localize("common.close")} | ||||||
|         </mwc-button> |         </ha-button> | ||||||
|       </ha-dialog> |       </ha-dialog> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| @@ -188,19 +193,14 @@ class HassioRepositoriesDialog extends LitElement { | |||||||
|         } |         } | ||||||
|         .option { |         .option { | ||||||
|           border: 1px solid var(--divider-color); |           border: 1px solid var(--divider-color); | ||||||
|           border-radius: 4px; |           border-radius: var(--ha-border-radius-sm); | ||||||
|           margin-top: 4px; |           margin-top: 4px; | ||||||
|         } |         } | ||||||
|         mwc-button { |         ha-button { | ||||||
|           margin-left: 8px; |           margin-left: 8px; | ||||||
|           margin-inline-start: 8px; |           margin-inline-start: 8px; | ||||||
|           margin-inline-end: initial; |           margin-inline-end: initial; | ||||||
|         } |         } | ||||||
|         ha-spinner { |  | ||||||
|           display: block; |  | ||||||
|           margin: 32px; |  | ||||||
|           text-align: center; |  | ||||||
|         } |  | ||||||
|         div.delete ha-icon-button { |         div.delete ha-icon-button { | ||||||
|           color: var(--error-color); |           color: var(--error-color); | ||||||
|         } |         } | ||||||
| @@ -249,6 +249,8 @@ class HassioRepositoriesDialog extends LitElement { | |||||||
|       await addStoreRepository(this.hass, input.value); |       await addStoreRepository(this.hass, input.value); | ||||||
|       await this._loadData(); |       await this._loadData(); | ||||||
|  |  | ||||||
|  |       fireEvent(this, "supervisor-collection-refresh", { collection: "store" }); | ||||||
|  |  | ||||||
|       input.value = ""; |       input.value = ""; | ||||||
|     } catch (err: any) { |     } catch (err: any) { | ||||||
|       this._error = extractApiErrorMessage(err); |       this._error = extractApiErrorMessage(err); | ||||||
| @@ -261,6 +263,8 @@ class HassioRepositoriesDialog extends LitElement { | |||||||
|     try { |     try { | ||||||
|       await removeStoreRepository(this.hass, slug); |       await removeStoreRepository(this.hass, slug); | ||||||
|       await this._loadData(); |       await this._loadData(); | ||||||
|  |  | ||||||
|  |       fireEvent(this, "supervisor-collection-refresh", { collection: "store" }); | ||||||
|     } catch (err: any) { |     } catch (err: any) { | ||||||
|       this._error = extractApiErrorMessage(err); |       this._error = extractApiErrorMessage(err); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -159,7 +159,7 @@ class HassioSystemManagedDialog extends LitElement { | |||||||
|           display: flex; |           display: flex; | ||||||
|           justify-content: center; |           justify-content: center; | ||||||
|           align-items: center; |           align-items: center; | ||||||
|           gap: 16px; |           gap: var(--ha-space-4); | ||||||
|           --mdc-icon-size: 48px; |           --mdc-icon-size: 48px; | ||||||
|           margin-bottom: 32px; |           margin-bottom: 32px; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ import type { PropertyValues, TemplateResult } from "lit"; | |||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; | import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||||
| import { navigate } from "../../../src/common/navigate"; | import { goBack, navigate } from "../../../src/common/navigate"; | ||||||
| import { extractSearchParam } from "../../../src/common/url/search-params"; | import { extractSearchParam } from "../../../src/common/url/search-params"; | ||||||
| import { nextRender } from "../../../src/common/util/render-status"; | import { nextRender } from "../../../src/common/util/render-status"; | ||||||
| import "../../../src/components/ha-icon-button"; | import "../../../src/components/ha-icon-button"; | ||||||
| @@ -193,7 +193,7 @@ class HassioIngressView extends LitElement { | |||||||
|         title: addon.name, |         title: addon.name, | ||||||
|       }); |       }); | ||||||
|       await nextRender(); |       await nextRender(); | ||||||
|       history.back(); |       goBack(); | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -275,7 +275,7 @@ class HassioIngressView extends LitElement { | |||||||
|         title: addon.name, |         title: addon.name, | ||||||
|       }); |       }); | ||||||
|       await nextRender(); |       await nextRender(); | ||||||
|       history.back(); |       goBack(); | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ export const hassioStyle = css` | |||||||
|   .card-group { |   .card-group { | ||||||
|     display: grid; |     display: grid; | ||||||
|     grid-template-columns: repeat(auto-fit, minmax(300px, 1fr)); |     grid-template-columns: repeat(auto-fit, minmax(300px, 1fr)); | ||||||
|     grid-gap: 8px; |     grid-gap: var(--ha-space-2); | ||||||
|   } |   } | ||||||
|   @media screen and (min-width: 640px) { |   @media screen and (min-width: 640px) { | ||||||
|     .card-group { |     .card-group { | ||||||
|   | |||||||
| @@ -1,10 +1,9 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
|  |  | ||||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | import type { CSSResultGroup, TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
| import { atLeastVersion } from "../../../src/common/config/version"; | import { atLeastVersion } from "../../../src/common/config/version"; | ||||||
| import "../../../src/components/buttons/ha-progress-button"; | import "../../../src/components/buttons/ha-progress-button"; | ||||||
|  | import "../../../src/components/ha-button"; | ||||||
| import "../../../src/components/ha-button-menu"; | import "../../../src/components/ha-button-menu"; | ||||||
| import "../../../src/components/ha-card"; | import "../../../src/components/ha-card"; | ||||||
| import "../../../src/components/ha-settings-row"; | import "../../../src/components/ha-settings-row"; | ||||||
| @@ -70,12 +69,12 @@ class HassioCoreInfo extends LitElement { | |||||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && |               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||||
|               this.supervisor.core.update_available |               this.supervisor.core.update_available | ||||||
|                 ? html` |                 ? html` | ||||||
|                     <a href="/hassio/update-available/core"> |                     <ha-button | ||||||
|                       <mwc-button |                       appearance="plain" | ||||||
|                         .label=${this.supervisor.localize("common.show")} |                       href="/hassio/update-available/core" | ||||||
|                       > |                     > | ||||||
|                       </mwc-button> |                       ${this.supervisor.localize("common.show")} | ||||||
|                     </a> |                     </ha-button> | ||||||
|                   ` |                   ` | ||||||
|                 : ""} |                 : ""} | ||||||
|             </ha-settings-row> |             </ha-settings-row> | ||||||
| @@ -95,7 +94,7 @@ class HassioCoreInfo extends LitElement { | |||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           <ha-progress-button |           <ha-progress-button | ||||||
|             slot="primaryAction" |             slot="primaryAction" | ||||||
|             class="warning" |             variant="danger" | ||||||
|             @click=${this._coreRestart} |             @click=${this._coreRestart} | ||||||
|             .title=${this.supervisor.localize("common.restart_name", { |             .title=${this.supervisor.localize("common.restart_name", { | ||||||
|               name: "Core", |               name: "Core", | ||||||
| @@ -188,11 +187,6 @@ class HassioCoreInfo extends LitElement { | |||||||
|           white-space: normal; |           white-space: normal; | ||||||
|           color: var(--secondary-text-color); |           color: var(--secondary-text-color); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .warning { |  | ||||||
|           --mdc-theme-primary: var(--error-color); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         ha-button-menu { |         ha-button-menu { | ||||||
|           color: var(--secondary-text-color); |           color: var(--secondary-text-color); | ||||||
|           --mdc-menu-min-width: 200px; |           --mdc-menu-min-width: 200px; | ||||||
|   | |||||||
| @@ -1,5 +1,3 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
|  |  | ||||||
| import { mdiDotsVertical } from "@mdi/js"; | import { mdiDotsVertical } from "@mdi/js"; | ||||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | import type { CSSResultGroup, TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| @@ -8,10 +6,11 @@ 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 "../../../src/components/buttons/ha-progress-button"; | import "../../../src/components/buttons/ha-progress-button"; | ||||||
|  | import "../../../src/components/ha-button"; | ||||||
| import "../../../src/components/ha-button-menu"; | import "../../../src/components/ha-button-menu"; | ||||||
| import "../../../src/components/ha-card"; | import "../../../src/components/ha-card"; | ||||||
| import "../../../src/components/ha-list-item"; |  | ||||||
| import "../../../src/components/ha-icon-button"; | import "../../../src/components/ha-icon-button"; | ||||||
|  | import "../../../src/components/ha-list-item"; | ||||||
| import "../../../src/components/ha-settings-row"; | import "../../../src/components/ha-settings-row"; | ||||||
| import { | import { | ||||||
|   extractApiErrorMessage, |   extractApiErrorMessage, | ||||||
| @@ -77,24 +76,28 @@ class HassioHostInfo extends LitElement { | |||||||
|                   <span slot="description"> |                   <span slot="description"> | ||||||
|                     ${this.supervisor.host.hostname} |                     ${this.supervisor.host.hostname} | ||||||
|                   </span> |                   </span> | ||||||
|                   <mwc-button |                   <ha-button | ||||||
|                     .label=${this.supervisor.localize("system.host.change")} |  | ||||||
|                     @click=${this._changeHostnameClicked} |                     @click=${this._changeHostnameClicked} | ||||||
|  |                     appearance="plain" | ||||||
|  |                     size="small" | ||||||
|                   > |                   > | ||||||
|                   </mwc-button> |                     ${this.supervisor.localize("system.host.change")} | ||||||
|  |                   </ha-button> | ||||||
|                 </ha-settings-row>` |                 </ha-settings-row>` | ||||||
|               : ""} |               : ""} | ||||||
|             ${this.supervisor.host.features.includes("network") |             ${this.supervisor.host.features.includes("network") | ||||||
|               ? html` <ha-settings-row> |               ? html`<ha-settings-row> | ||||||
|                   <span slot="heading"> |                   <span slot="heading"> | ||||||
|                     ${this.supervisor.localize("system.host.ip_address")} |                     ${this.supervisor.localize("system.host.ip_address")} | ||||||
|                   </span> |                   </span> | ||||||
|                   <span slot="description"> ${primaryIpAddress} </span> |                   <span slot="description"> ${primaryIpAddress} </span> | ||||||
|                   <mwc-button |                   <ha-button | ||||||
|                     .label=${this.supervisor.localize("system.host.change")} |  | ||||||
|                     @click=${this._changeNetworkClicked} |                     @click=${this._changeNetworkClicked} | ||||||
|  |                     appearance="plain" | ||||||
|  |                     size="small" | ||||||
|                   > |                   > | ||||||
|                   </mwc-button> |                     ${this.supervisor.localize("system.host.change")} | ||||||
|  |                   </ha-button> | ||||||
|                 </ha-settings-row>` |                 </ha-settings-row>` | ||||||
|               : ""} |               : ""} | ||||||
|  |  | ||||||
| @@ -108,12 +111,13 @@ class HassioHostInfo extends LitElement { | |||||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && |               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||||
|               this.supervisor.os.update_available |               this.supervisor.os.update_available | ||||||
|                 ? html` |                 ? html` | ||||||
|                     <a href="/hassio/update-available/os"> |                     <ha-button | ||||||
|                       <mwc-button |                       appearance="plain" | ||||||
|                         .label=${this.supervisor.localize("common.show")} |                       size="small" | ||||||
|                       > |                       href="/hassio/update-available/os" | ||||||
|                       </mwc-button> |                     > | ||||||
|                     </a> |                       ${this.supervisor.localize("common.show")} | ||||||
|  |                     </ha-button> | ||||||
|                   ` |                   ` | ||||||
|                 : ""} |                 : ""} | ||||||
|             </ha-settings-row> |             </ha-settings-row> | ||||||
| @@ -139,16 +143,12 @@ class HassioHostInfo extends LitElement { | |||||||
|               : ""} |               : ""} | ||||||
|           </div> |           </div> | ||||||
|           <div> |           <div> | ||||||
|             ${this.supervisor.host.disk_life_time !== "" && |             ${this.supervisor.host.disk_life_time !== null | ||||||
|             this.supervisor.host.disk_life_time >= 10 |  | ||||||
|               ? html` <ha-settings-row> |               ? html` <ha-settings-row> | ||||||
|                   <span slot="heading"> |                   <span slot="heading"> | ||||||
|                     ${this.supervisor.localize( |                     ${this.supervisor.localize("system.host.lifetime_used")} | ||||||
|                       "system.host.emmc_lifetime_used" |  | ||||||
|                     )} |  | ||||||
|                   </span> |                   </span> | ||||||
|                   <span slot="description"> |                   <span slot="description"> | ||||||
|                     ${this.supervisor.host.disk_life_time - 10} % - |  | ||||||
|                     ${this.supervisor.host.disk_life_time} % |                     ${this.supervisor.host.disk_life_time} % | ||||||
|                   </span> |                   </span> | ||||||
|                 </ha-settings-row>` |                 </ha-settings-row>` | ||||||
| @@ -167,7 +167,7 @@ class HassioHostInfo extends LitElement { | |||||||
|         <div class="card-actions"> |         <div class="card-actions"> | ||||||
|           ${this.supervisor.host.features.includes("reboot") |           ${this.supervisor.host.features.includes("reboot") | ||||||
|             ? html` |             ? html` | ||||||
|                 <ha-progress-button class="warning" @click=${this._hostReboot}> |                 <ha-progress-button variant="danger" @click=${this._hostReboot}> | ||||||
|                   ${this.supervisor.localize("system.host.reboot_host")} |                   ${this.supervisor.localize("system.host.reboot_host")} | ||||||
|                 </ha-progress-button> |                 </ha-progress-button> | ||||||
|               ` |               ` | ||||||
| @@ -175,7 +175,7 @@ class HassioHostInfo extends LitElement { | |||||||
|           ${this.supervisor.host.features.includes("shutdown") |           ${this.supervisor.host.features.includes("shutdown") | ||||||
|             ? html` |             ? html` | ||||||
|                 <ha-progress-button |                 <ha-progress-button | ||||||
|                   class="warning" |                   variant="danger" | ||||||
|                   @click=${this._hostShutdown} |                   @click=${this._hostShutdown} | ||||||
|                 > |                 > | ||||||
|                   ${this.supervisor.localize("system.host.shutdown_host")} |                   ${this.supervisor.localize("system.host.shutdown_host")} | ||||||
| @@ -431,10 +431,6 @@ class HassioHostInfo extends LitElement { | |||||||
|           color: var(--secondary-text-color); |           color: var(--secondary-text-color); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         .warning { |  | ||||||
|           --mdc-theme-primary: var(--error-color); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         ha-button-menu { |         ha-button-menu { | ||||||
|           color: var(--secondary-text-color); |           color: var(--secondary-text-color); | ||||||
|           --mdc-menu-min-width: 200px; |           --mdc-menu-min-width: 200px; | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ import { atLeastVersion } from "../../../src/common/config/version"; | |||||||
| 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-button"; | ||||||
| import "../../../src/components/ha-card"; | import "../../../src/components/ha-card"; | ||||||
| import "../../../src/components/ha-settings-row"; | import "../../../src/components/ha-settings-row"; | ||||||
| import "../../../src/components/ha-switch"; | import "../../../src/components/ha-switch"; | ||||||
| @@ -80,12 +81,13 @@ class HassioSupervisorInfo extends LitElement { | |||||||
|               ${!atLeastVersion(this.hass.config.version, 2021, 12) && |               ${!atLeastVersion(this.hass.config.version, 2021, 12) && | ||||||
|               this.supervisor.supervisor.update_available |               this.supervisor.supervisor.update_available | ||||||
|                 ? html` |                 ? html` | ||||||
|                     <a href="/hassio/update-available/supervisor"> |                     <ha-button | ||||||
|                       <mwc-button |                       appearance="plain" | ||||||
|                         .label=${this.supervisor.localize("common.show")} |                       size="small" | ||||||
|                       > |                       href="/hassio/update-available/supervisor" | ||||||
|                       </mwc-button> |                     > | ||||||
|                     </a> |                       ${this.supervisor.localize("common.show")} | ||||||
|  |                     </ha-button> | ||||||
|                   ` |                   ` | ||||||
|                 : ""} |                 : ""} | ||||||
|             </ha-settings-row> |             </ha-settings-row> | ||||||
| @@ -156,24 +158,28 @@ class HassioSupervisorInfo extends LitElement { | |||||||
|                   ${this.supervisor.localize( |                   ${this.supervisor.localize( | ||||||
|                     "system.supervisor.unsupported_title" |                     "system.supervisor.unsupported_title" | ||||||
|                   )} |                   )} | ||||||
|                   <mwc-button |                   <ha-button | ||||||
|                     slot="action" |                     slot="action" | ||||||
|                     .label=${this.supervisor.localize("common.learn_more")} |  | ||||||
|                     @click=${this._unsupportedDialog} |                     @click=${this._unsupportedDialog} | ||||||
|  |                     variant="warning" | ||||||
|  |                     size="small" | ||||||
|                   > |                   > | ||||||
|                   </mwc-button> |                     ${this.supervisor.localize("common.learn_more")} | ||||||
|  |                   </ha-button> | ||||||
|                 </ha-alert>`} |                 </ha-alert>`} | ||||||
|             ${!this.supervisor.supervisor.healthy |             ${!this.supervisor.supervisor.healthy | ||||||
|               ? html`<ha-alert alert-type="error"> |               ? html`<ha-alert alert-type="error"> | ||||||
|                   ${this.supervisor.localize( |                   ${this.supervisor.localize( | ||||||
|                     "system.supervisor.unhealthy_title" |                     "system.supervisor.unhealthy_title" | ||||||
|                   )} |                   )} | ||||||
|                   <mwc-button |                   <ha-button | ||||||
|  |                     variant="danger" | ||||||
|  |                     size="small" | ||||||
|                     slot="action" |                     slot="action" | ||||||
|                     .label=${this.supervisor.localize("common.learn_more")} |  | ||||||
|                     @click=${this._unhealthyDialog} |                     @click=${this._unhealthyDialog} | ||||||
|                   > |                   > | ||||||
|                   </mwc-button> |                     ${this.supervisor.localize("common.learn_more")} | ||||||
|  |                   </ha-button> | ||||||
|                 </ha-alert>` |                 </ha-alert>` | ||||||
|               : ""} |               : ""} | ||||||
|           </div> |           </div> | ||||||
| @@ -448,9 +454,6 @@ class HassioSupervisorInfo extends LitElement { | |||||||
|           white-space: normal; |           white-space: normal; | ||||||
|           color: var(--secondary-text-color); |           color: var(--secondary-text-color); | ||||||
|         } |         } | ||||||
|         ha-alert mwc-button { |  | ||||||
|           --mdc-theme-primary: var(--primary-text-color); |  | ||||||
|         } |  | ||||||
|         a { |         a { | ||||||
|           text-decoration: none; |           text-decoration: none; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -1,5 +1,3 @@ | |||||||
| import "@material/mwc-button"; |  | ||||||
|  |  | ||||||
| import type { CSSResultGroup, TemplateResult } from "lit"; | import type { CSSResultGroup, TemplateResult } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement, property, state } from "lit/decorators"; | import { customElement, property, state } from "lit/decorators"; | ||||||
|   | |||||||
| @@ -208,14 +208,16 @@ class UpdateAvailableCard extends LitElement { | |||||||
|               <div class="card-actions"> |               <div class="card-actions"> | ||||||
|                 ${changelog |                 ${changelog | ||||||
|                   ? html` |                   ? html` | ||||||
|                       <a href=${changelog} target="_blank" rel="noreferrer"> |                       <ha-button | ||||||
|                         <ha-button |                         href=${changelog} | ||||||
|                           .label=${this.supervisor.localize( |                         target="_blank" | ||||||
|                             "update_available.open_release_notes" |                         rel="noreferrer" | ||||||
|                           )} |                         appearance="plain" | ||||||
|                         > |                       > | ||||||
|                         </ha-button> |                         ${this.supervisor.localize( | ||||||
|                       </a> |                           "update_available.open_release_notes" | ||||||
|  |                         )} | ||||||
|  |                       </ha-button> | ||||||
|                     ` |                     ` | ||||||
|                   : nothing} |                   : nothing} | ||||||
|                 <span></span> |                 <span></span> | ||||||
|   | |||||||
| @@ -2,6 +2,7 @@ import type { TemplateResult } from "lit"; | |||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement } from "lit"; | ||||||
| import { customElement, property } from "lit/decorators"; | import { customElement, property } from "lit/decorators"; | ||||||
| import type { Supervisor } from "../../../src/data/supervisor/supervisor"; | import type { Supervisor } from "../../../src/data/supervisor/supervisor"; | ||||||
|  | import { goBack } from "../../../src/common/navigate"; | ||||||
| import "../../../src/layouts/hass-subpage"; | import "../../../src/layouts/hass-subpage"; | ||||||
| import type { HomeAssistant, Route } from "../../../src/types"; | import type { HomeAssistant, Route } from "../../../src/types"; | ||||||
| import "./update-available-card"; | import "./update-available-card"; | ||||||
| @@ -35,7 +36,7 @@ class UpdateAvailableDashboard extends LitElement { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _updateComplete() { |   private _updateComplete() { | ||||||
|     history.back(); |     goBack(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static styles = css` |   static styles = css` | ||||||
|   | |||||||
| @@ -3,26 +3,26 @@ import { mdiArrowCollapseDown, mdiDownload } from "@mdi/js"; | |||||||
| // eslint-disable-next-line import/extensions | // eslint-disable-next-line import/extensions | ||||||
| import { IntersectionController } from "@lit-labs/observers/intersection-controller.js"; | import { IntersectionController } from "@lit-labs/observers/intersection-controller.js"; | ||||||
| import { LitElement, type PropertyValues, css, html, nothing } from "lit"; | import { LitElement, type PropertyValues, css, html, nothing } from "lit"; | ||||||
| import { classMap } from "lit/directives/class-map"; |  | ||||||
| import { customElement, property, query, state } from "lit/decorators"; | import { customElement, property, query, state } from "lit/decorators"; | ||||||
|  | import { classMap } from "lit/directives/class-map"; | ||||||
|  | import { fireEvent } from "../../../src/common/dom/fire_event"; | ||||||
| import type { | import type { | ||||||
|   LandingPageKeys, |   LandingPageKeys, | ||||||
|   LocalizeFunc, |   LocalizeFunc, | ||||||
| } from "../../../src/common/translations/localize"; | } from "../../../src/common/translations/localize"; | ||||||
|  | import { waitForSeconds } from "../../../src/common/util/wait"; | ||||||
|  | import "../../../src/components/ha-alert"; | ||||||
|  | import "../../../src/components/ha-ansi-to-html"; | ||||||
|  | import type { HaAnsiToHtml } from "../../../src/components/ha-ansi-to-html"; | ||||||
| import "../../../src/components/ha-button"; | import "../../../src/components/ha-button"; | ||||||
| import "../../../src/components/ha-icon-button"; | import "../../../src/components/ha-icon-button"; | ||||||
| import "../../../src/components/ha-svg-icon"; | import "../../../src/components/ha-svg-icon"; | ||||||
| import "../../../src/components/ha-ansi-to-html"; | import { fileDownload } from "../../../src/util/file_download"; | ||||||
| import "../../../src/components/ha-alert"; |  | ||||||
| import type { HaAnsiToHtml } from "../../../src/components/ha-ansi-to-html"; |  | ||||||
| import { | import { | ||||||
|   getObserverLogs, |   getObserverLogs, | ||||||
|   downloadUrl as observerLogsDownloadUrl, |   downloadUrl as observerLogsDownloadUrl, | ||||||
| } from "../data/observer"; | } from "../data/observer"; | ||||||
| import { fireEvent } from "../../../src/common/dom/fire_event"; |  | ||||||
| import { fileDownload } from "../../../src/util/file_download"; |  | ||||||
| import { getSupervisorLogs, getSupervisorLogsFollow } from "../data/supervisor"; | import { getSupervisorLogs, getSupervisorLogsFollow } from "../data/supervisor"; | ||||||
| import { waitForSeconds } from "../../../src/common/util/wait"; |  | ||||||
| import { ASSUME_CORE_START_SECONDS } from "../ha-landing-page"; | import { ASSUME_CORE_START_SECONDS } from "../ha-landing-page"; | ||||||
|  |  | ||||||
| const ERROR_CHECK = /^[\d\s-:]+(ERROR|CRITICAL)(.*)/gm; | const ERROR_CHECK = /^[\d\s-:]+(ERROR|CRITICAL)(.*)/gm; | ||||||
| @@ -64,7 +64,7 @@ class LandingPageLogs extends LitElement { | |||||||
|   protected render() { |   protected render() { | ||||||
|     return html` |     return html` | ||||||
|       <div class="actions"> |       <div class="actions"> | ||||||
|         <ha-button @click=${this._toggleLogDetails}> |         <ha-button appearance="plain" @click=${this._toggleLogDetails}> | ||||||
|           ${this.localize(this._show ? "hide_details" : "show_details")} |           ${this.localize(this._show ? "hide_details" : "show_details")} | ||||||
|         </ha-button> |         </ha-button> | ||||||
|         ${this._show |         ${this._show | ||||||
| @@ -81,7 +81,11 @@ class LandingPageLogs extends LitElement { | |||||||
|               alert-type="error" |               alert-type="error" | ||||||
|               .title=${this.localize("logs.fetch_error")} |               .title=${this.localize("logs.fetch_error")} | ||||||
|             > |             > | ||||||
|               <ha-button @click=${this._startLogStream}> |               <ha-button | ||||||
|  |                 size="small" | ||||||
|  |                 variant="danger" | ||||||
|  |                 @click=${this._startLogStream} | ||||||
|  |               > | ||||||
|                 ${this.localize("logs.retry")} |                 ${this.localize("logs.retry")} | ||||||
|               </ha-button> |               </ha-button> | ||||||
|             </ha-alert> |             </ha-alert> | ||||||
| @@ -104,14 +108,13 @@ class LandingPageLogs extends LitElement { | |||||||
|               !this._scrolledToBottomController.value) || |               !this._scrolledToBottomController.value) || | ||||||
|             false, |             false, | ||||||
|         })}" |         })}" | ||||||
|  |         size="small" | ||||||
|  |         appearance="filled" | ||||||
|         @click=${this._scrollToBottom} |         @click=${this._scrollToBottom} | ||||||
|       > |       > | ||||||
|         <ha-svg-icon .path=${mdiArrowCollapseDown} slot="icon"></ha-svg-icon> |         <ha-svg-icon .path=${mdiArrowCollapseDown} slot="start"></ha-svg-icon> | ||||||
|         ${this.localize("logs.scroll_down_button")} |         ${this.localize("logs.scroll_down_button")} | ||||||
|         <ha-svg-icon |         <ha-svg-icon .path=${mdiArrowCollapseDown} slot="end"></ha-svg-icon> | ||||||
|           .path=${mdiArrowCollapseDown} |  | ||||||
|           slot="trailingIcon" |  | ||||||
|         ></ha-svg-icon> |  | ||||||
|       </ha-button> |       </ha-button> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| @@ -299,7 +302,7 @@ class LandingPageLogs extends LitElement { | |||||||
|         max-height: 300px; |         max-height: 300px; | ||||||
|         overflow: auto; |         overflow: auto; | ||||||
|         border: 1px solid var(--divider-color); |         border: 1px solid var(--divider-color); | ||||||
|         border-radius: 4px; |         border-radius: var(--ha-border-radius-sm); | ||||||
|         padding: 4px; |         padding: 4px; | ||||||
|       } |       } | ||||||
|  |  | ||||||
| @@ -308,21 +311,14 @@ class LandingPageLogs extends LitElement { | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       .new-logs-indicator { |       .new-logs-indicator { | ||||||
|         --mdc-theme-primary: var(--text-primary-color); |  | ||||||
|  |  | ||||||
|         overflow: hidden; |         overflow: hidden; | ||||||
|         position: absolute; |         position: absolute; | ||||||
|         bottom: 0; |         bottom: 4px; | ||||||
|         left: 0; |         left: 4px; | ||||||
|         right: 0; |  | ||||||
|         height: 0; |         height: 0; | ||||||
|         background-color: var(--primary-color); |  | ||||||
|         border-radius: 8px; |  | ||||||
|  |  | ||||||
|         transition: height 0.4s ease-out; |         transition: height 0.4s ease-out; | ||||||
|         display: flex; |         display: flex; | ||||||
|         justify-content: space-between; |  | ||||||
|         align-items: center; |  | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .new-logs-indicator.visible { |       .new-logs-indicator.visible { | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user