mirror of
				https://github.com/home-assistant/frontend.git
				synced 2025-10-25 11:39:41 +00:00 
			
		
		
		
	Compare commits
	
		
			438 Commits
		
	
	
		
			20250903.2
			...
			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 | ||
|   | dcbad9e798 | ||
|   | 26b3212c7e | ||
|   | f3f4bcfe45 | ||
|   | 7cfa9de75f | ||
|   | b66dc8894d | ||
|   | 14a7813ab0 | ||
|   | 70a2ca281f | ||
|   | d982f042fc | ||
|   | e60f9e326b | ||
|   | ba39d189e7 | ||
|   | 78867b2cd9 | ||
|   | 1dff42dc00 | ||
|   | 0c9b3a0765 | ||
|   | 5a109c0ba8 | ||
|   | f3b214c30a | ||
|   | c49d2a0be6 | ||
|   | c6c3170c1b | ||
|   | 0abb958aea | ||
|   | 9d55843629 | ||
|   | b70d309297 | ||
|   | 5961b71562 | ||
|   | 6942626f60 | ||
|   | 069c0acdff | ||
|   | 1f0d83190d | ||
|   | 7c6c92c856 | ||
|   | 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 | ||
|   | 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 | ||
|   | 25f25243bd | ||
|   | cf8d36b1f3 | ||
|   | e3a9d754df | ||
|   | 7b303a699b | ||
| ![renovate[bot]](/assets/img/avatar_default.png)  | ee45eb00f7 | ||
|   | 24a6aa2669 | ||
|   | 66d011cfb9 | ||
|   | 35895735cc | ||
|   | e71df0b71a | ||
|   | 2a9846c598 | ||
|   | b243d56bee | ||
|   | 6a372a165e | ||
|   | a5dad9bc22 | 
							
								
								
									
										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@v5.0.0 |         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@v5.0.0 |         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@v5.0.0 |         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.4 |         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@v5.0.0 |         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@v5.0.0 |         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@v5.0.0 |         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@v5.0.0 |         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@v5.0.0 |         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@v5.0.0 |         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@v5.0.0 |         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@v5.0.0 |         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@v5.0.0 |         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.1 |         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@v5.0.0 |         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.07.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@v5.0.0 |         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@v5.0.0 |         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 | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/restrict-task-creation.yml
									
									
									
									
										vendored
									
									
								
							| @@ -12,7 +12,7 @@ jobs: | |||||||
|     if: github.event.issue.type.name == 'Task' |     if: github.event.issue.type.name == 'Task' | ||||||
|     steps: |     steps: | ||||||
|       - name: Check if user is authorized |       - name: Check if user is authorized | ||||||
|         uses: actions/github-script@v7 |         uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8 | ||||||
|         with: |         with: | ||||||
|           script: | |           script: | | ||||||
|             const issueAuthor = context.payload.issue.user.login; |             const issueAuthor = context.payload.issue.user.login; | ||||||
|   | |||||||
							
								
								
									
										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 | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/translations.yaml
									
									
									
									
										vendored
									
									
								
							| @@ -14,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@v5.0.0 |         uses: actions/checkout@08c6903cd8c0fde910a37f88322edcfb5dd907a8 # v5.0.0 | ||||||
|  |  | ||||||
|       - name: Upload Translations |       - name: Upload Translations | ||||||
|         run: | |         run: | | ||||||
|   | |||||||
										
											
												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.3.cjs | yarnPath: .yarn/releases/yarn-4.10.3.cjs | ||||||
|   | |||||||
| @@ -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}/`)), | ||||||
|     }, |     }, | ||||||
|   | |||||||
| @@ -242,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; | ||||||
|   | |||||||
| @@ -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", | ||||||
|               }, |               }, | ||||||
|             ], |             ], | ||||||
|           }, |           }, | ||||||
|   | |||||||
| @@ -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', | ||||||
|   | |||||||
| @@ -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, 0px), 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 { | ||||||
|   | |||||||
| @@ -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) | ||||||
|  |               ? html`<state-card-content | ||||||
|                   .stateObj=${state} |                   .stateObj=${state} | ||||||
|                   .hass=${this.hass} |                   .hass=${this.hass} | ||||||
|                   in-dialog |                   in-dialog | ||||||
|             ></state-card-content> |                 ></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); | ||||||
|       } |       } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -117,7 +117,7 @@ export class DemoHaBadge extends LitElement { | |||||||
|     } |     } | ||||||
|     .card-content { |     .card-content { | ||||||
|       display: flex; |       display: flex; | ||||||
|       gap: 24px; |       gap: var(--ha-space-6); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -155,11 +155,11 @@ export class DemoHaButton extends LitElement { | |||||||
|     .card-content { |     .card-content { | ||||||
|       display: flex; |       display: flex; | ||||||
|       flex-direction: column; |       flex-direction: column; | ||||||
|       gap: 24px; |       gap: var(--ha-space-6); | ||||||
|     } |     } | ||||||
|     .card-content div { |     .card-content div { | ||||||
|       display: flex; |       display: flex; | ||||||
|       gap: 8px; |       gap: var(--ha-space-2); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -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; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -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; | ||||||
| @@ -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; | ||||||
| @@ -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; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										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; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -123,11 +123,11 @@ export class DemoHaProgressButton extends LitElement { | |||||||
|     .card-content { |     .card-content { | ||||||
|       display: flex; |       display: flex; | ||||||
|       flex-direction: column; |       flex-direction: column; | ||||||
|       gap: 24px; |       gap: var(--ha-space-6); | ||||||
|     } |     } | ||||||
|     .card-content div { |     .card-content div { | ||||||
|       display: flex; |       display: flex; | ||||||
|       gap: 8px; |       gap: var(--ha-space-2); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|   | |||||||
| @@ -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 { | ||||||
|   | |||||||
							
								
								
									
										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; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -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); | ||||||
|     } |     } | ||||||
|   `; |   `; | ||||||
|   | |||||||
| @@ -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,41 +82,83 @@ 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" |   ); | ||||||
|           ? { |  | ||||||
|  |   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, |         name: entry.name, | ||||||
|         required: entry.required, |         required: entry.required, | ||||||
|               selector: { select: { options: entry.options } }, |         selector, | ||||||
|  |       }; | ||||||
|     } |     } | ||||||
|           : entry.type === "string" |     return entry; | ||||||
|             ? entry.multiple |  | ||||||
|               ? { |  | ||||||
|                   name: entry.name, |  | ||||||
|                   required: entry.required, |  | ||||||
|                   selector: { |  | ||||||
|                     select: { options: [], multiple: true, custom_value: true }, |  | ||||||
|                   }, |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   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 } } | ||||||
|         : { |         : { | ||||||
|                   name: entry.name, |  | ||||||
|                   required: entry.required, |  | ||||||
|                   selector: { |  | ||||||
|             text: { |             text: { | ||||||
|               type: entry.format |               type: entry.format | ||||||
|                 ? entry.format |                 ? entry.format | ||||||
| @@ -120,36 +166,41 @@ class HassioAddonConfig extends LitElement { | |||||||
|                   ? "password" |                   ? "password" | ||||||
|                   : "text", |                   : "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, | ||||||
|         }, |         }, | ||||||
|  |       }; | ||||||
|     } |     } | ||||||
|             : entry.type === "boolean" |     if (entry.type === "float" || entry.type === "integer") { | ||||||
|               ? { |       return { | ||||||
|                   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: { |         number: { | ||||||
|           mode: "box", |           mode: "box", | ||||||
|           step: entry.type === "float" ? "any" : undefined, |           step: entry.type === "float" ? "any" : undefined, | ||||||
|         }, |         }, | ||||||
|                       }, |       }; | ||||||
|  |     } | ||||||
|  |     if (force) { | ||||||
|  |       return { object: {} }; | ||||||
|  |     } | ||||||
|  |     return null; | ||||||
|   } |   } | ||||||
|                   : entry |  | ||||||
|       ) |  | ||||||
|   ); |  | ||||||
|  |  | ||||||
|   private _filteredShchema = memoizeOne( |   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! | ||||||
|                       ) |                       ) | ||||||
|   | |||||||
| @@ -781,7 +781,7 @@ class HassioAddonInfo extends LitElement { | |||||||
|  |  | ||||||
|       ${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} | ||||||
| @@ -1333,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); | ||||||
|   | |||||||
| @@ -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; | ||||||
|   | |||||||
| @@ -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; | ||||||
|   | |||||||
| @@ -15,6 +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-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"; | ||||||
| @@ -36,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"]; | ||||||
|  |  | ||||||
| @@ -114,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())} | ||||||
| @@ -627,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; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -228,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 { | ||||||
|   | |||||||
| @@ -119,15 +119,17 @@ 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" | ||||||
|                         )} |                         )} | ||||||
|                       > |                       </ha-tooltip> | ||||||
|                         <div> |                       <div .id="icon-button-${repo.slug}"> | ||||||
|                         <ha-icon-button |                         <ha-icon-button | ||||||
|                           .disabled=${usedRepositories.includes(repo.slug)} |                           .disabled=${usedRepositories.includes(repo.slug)} | ||||||
|                           .slug=${repo.slug} |                           .slug=${repo.slug} | ||||||
| @@ -138,7 +140,6 @@ class HassioRepositoriesDialog extends LitElement { | |||||||
|                         > |                         > | ||||||
|                         </ha-icon-button> |                         </ha-icon-button> | ||||||
|                       </div> |                       </div> | ||||||
|                       </ha-tooltip> |  | ||||||
|                     </ha-md-list-item> |                     </ha-md-list-item> | ||||||
|                   ` |                   ` | ||||||
|                 ) |                 ) | ||||||
| @@ -192,7 +193,7 @@ 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; | ||||||
|         } |         } | ||||||
|         ha-button { |         ha-button { | ||||||
|   | |||||||
| @@ -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 { | ||||||
|   | |||||||
| @@ -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` | ||||||
|   | |||||||
| @@ -302,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; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -213,7 +213,7 @@ class HaLandingPage extends LandingPageBaseElement { | |||||||
|       ha-card .card-content { |       ha-card .card-content { | ||||||
|         display: flex; |         display: flex; | ||||||
|         flex-direction: column; |         flex-direction: column; | ||||||
|         gap: 16px; |         gap: var(--ha-space-4); | ||||||
|       } |       } | ||||||
|       ha-alert p { |       ha-alert p { | ||||||
|         text-align: unset; |         text-align: unset; | ||||||
| @@ -221,7 +221,7 @@ class HaLandingPage extends LandingPageBaseElement { | |||||||
|       ha-language-picker { |       ha-language-picker { | ||||||
|         display: block; |         display: block; | ||||||
|         width: 200px; |         width: 200px; | ||||||
|         border-radius: 4px; |         border-radius: var(--ha-border-radius-sm); | ||||||
|         overflow: hidden; |         overflow: hidden; | ||||||
|         --ha-select-height: 40px; |         --ha-select-height: 40px; | ||||||
|         --mdc-select-fill-color: none; |         --mdc-select-fill-color: none; | ||||||
|   | |||||||
| @@ -19,8 +19,9 @@ | |||||||
|         height: auto; |         height: auto; | ||||||
|         padding: 32px 0; |         padding: 32px 0; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       .content { |       .content { | ||||||
|         max-width: 560px; |         max-width: min(560px, calc(100vw - var(--safe-area-inset-right, 0px) - var(--safe-area-inset-left, 0px))); | ||||||
|         margin: 0 auto; |         margin: 0 auto; | ||||||
|         padding: 0 16px; |         padding: 0 16px; | ||||||
|         box-sizing: content-box; |         box-sizing: content-box; | ||||||
|   | |||||||
							
								
								
									
										100
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										100
									
								
								package.json
									
									
									
									
									
								
							| @@ -26,32 +26,33 @@ | |||||||
|   "license": "Apache-2.0", |   "license": "Apache-2.0", | ||||||
|   "type": "module", |   "type": "module", | ||||||
|   "dependencies": { |   "dependencies": { | ||||||
|     "@awesome.me/webawesome": "3.0.0-beta.4", |     "@babel/runtime": "7.28.4", | ||||||
|     "@babel/runtime": "7.28.3", |  | ||||||
|     "@braintree/sanitize-url": "7.1.1", |     "@braintree/sanitize-url": "7.1.1", | ||||||
|     "@codemirror/autocomplete": "6.18.6", |     "@codemirror/autocomplete": "6.19.0", | ||||||
|     "@codemirror/commands": "6.8.1", |     "@codemirror/commands": "6.9.0", | ||||||
|     "@codemirror/language": "6.11.3", |     "@codemirror/language": "6.11.3", | ||||||
|     "@codemirror/legacy-modes": "6.5.1", |     "@codemirror/legacy-modes": "6.5.2", | ||||||
|     "@codemirror/search": "6.5.11", |     "@codemirror/search": "6.5.11", | ||||||
|     "@codemirror/state": "6.5.2", |     "@codemirror/state": "6.5.2", | ||||||
|     "@codemirror/view": "6.38.1", |     "@codemirror/view": "6.38.5", | ||||||
|  |     "@date-fns/tz": "1.4.1", | ||||||
|     "@egjs/hammerjs": "2.0.17", |     "@egjs/hammerjs": "2.0.17", | ||||||
|     "@formatjs/intl-datetimeformat": "6.18.0", |     "@formatjs/intl-datetimeformat": "6.18.1", | ||||||
|     "@formatjs/intl-displaynames": "6.8.11", |     "@formatjs/intl-displaynames": "6.8.12", | ||||||
|     "@formatjs/intl-durationformat": "0.7.4", |     "@formatjs/intl-durationformat": "0.7.5", | ||||||
|     "@formatjs/intl-getcanonicallocales": "2.5.5", |     "@formatjs/intl-getcanonicallocales": "2.5.6", | ||||||
|     "@formatjs/intl-listformat": "7.7.11", |     "@formatjs/intl-listformat": "7.7.12", | ||||||
|     "@formatjs/intl-locale": "4.2.11", |     "@formatjs/intl-locale": "4.2.12", | ||||||
|     "@formatjs/intl-numberformat": "8.15.4", |     "@formatjs/intl-numberformat": "8.15.5", | ||||||
|     "@formatjs/intl-pluralrules": "5.4.4", |     "@formatjs/intl-pluralrules": "5.4.5", | ||||||
|     "@formatjs/intl-relativetimeformat": "11.4.11", |     "@formatjs/intl-relativetimeformat": "11.4.12", | ||||||
|     "@fullcalendar/core": "6.1.19", |     "@fullcalendar/core": "6.1.19", | ||||||
|     "@fullcalendar/daygrid": "6.1.19", |     "@fullcalendar/daygrid": "6.1.19", | ||||||
|     "@fullcalendar/interaction": "6.1.19", |     "@fullcalendar/interaction": "6.1.19", | ||||||
|     "@fullcalendar/list": "6.1.19", |     "@fullcalendar/list": "6.1.19", | ||||||
|     "@fullcalendar/luxon3": "6.1.19", |     "@fullcalendar/luxon3": "6.1.19", | ||||||
|     "@fullcalendar/timegrid": "6.1.19", |     "@fullcalendar/timegrid": "6.1.19", | ||||||
|  |     "@home-assistant/webawesome": "3.0.0-beta.6.ha.4", | ||||||
|     "@lezer/highlight": "1.2.1", |     "@lezer/highlight": "1.2.1", | ||||||
|     "@lit-labs/motion": "1.0.9", |     "@lit-labs/motion": "1.0.9", | ||||||
|     "@lit-labs/observers": "2.0.6", |     "@lit-labs/observers": "2.0.6", | ||||||
| @@ -84,26 +85,24 @@ | |||||||
|     "@mdi/js": "7.4.47", |     "@mdi/js": "7.4.47", | ||||||
|     "@mdi/svg": "7.4.47", |     "@mdi/svg": "7.4.47", | ||||||
|     "@replit/codemirror-indentation-markers": "6.5.3", |     "@replit/codemirror-indentation-markers": "6.5.3", | ||||||
|     "@shoelace-style/shoelace": "2.20.1", |  | ||||||
|     "@swc/helpers": "0.5.17", |     "@swc/helpers": "0.5.17", | ||||||
|     "@thomasloven/round-slider": "0.6.0", |     "@thomasloven/round-slider": "0.6.0", | ||||||
|     "@tsparticles/engine": "3.9.1", |     "@tsparticles/engine": "3.9.1", | ||||||
|     "@tsparticles/preset-links": "3.2.0", |     "@tsparticles/preset-links": "3.2.0", | ||||||
|     "@vaadin/combo-box": "24.8.5", |     "@vaadin/combo-box": "24.9.2", | ||||||
|     "@vaadin/vaadin-themable-mixin": "24.8.5", |     "@vaadin/vaadin-themable-mixin": "24.9.2", | ||||||
|     "@vibrant/color": "4.0.0", |     "@vibrant/color": "4.0.0", | ||||||
|     "@vue/web-component-wrapper": "1.3.0", |     "@vue/web-component-wrapper": "1.3.0", | ||||||
|     "@webcomponents/scoped-custom-element-registry": "0.0.10", |     "@webcomponents/scoped-custom-element-registry": "0.0.10", | ||||||
|     "@webcomponents/webcomponentsjs": "2.8.0", |     "@webcomponents/webcomponentsjs": "2.8.0", | ||||||
|     "app-datepicker": "5.1.1", |     "app-datepicker": "5.1.1", | ||||||
|     "barcode-detector": "3.0.5", |     "barcode-detector": "3.0.6", | ||||||
|     "color-name": "2.0.0", |     "color-name": "2.0.2", | ||||||
|     "comlink": "4.4.2", |     "comlink": "4.4.2", | ||||||
|     "core-js": "3.45.1", |     "core-js": "3.45.1", | ||||||
|     "cropperjs": "1.6.2", |     "cropperjs": "1.6.2", | ||||||
|     "culori": "4.0.2", |     "culori": "4.0.2", | ||||||
|     "date-fns": "4.1.0", |     "date-fns": "4.1.0", | ||||||
|     "date-fns-tz": "3.2.0", |  | ||||||
|     "deep-clone-simple": "1.1.1", |     "deep-clone-simple": "1.1.1", | ||||||
|     "deep-freeze": "0.0.1", |     "deep-freeze": "0.0.1", | ||||||
|     "dialog-polyfill": "0.5.6", |     "dialog-polyfill": "0.5.6", | ||||||
| @@ -112,18 +111,18 @@ | |||||||
|     "fuse.js": "7.1.0", |     "fuse.js": "7.1.0", | ||||||
|     "google-timezones-json": "1.2.0", |     "google-timezones-json": "1.2.0", | ||||||
|     "gulp-zopfli-green": "6.0.2", |     "gulp-zopfli-green": "6.0.2", | ||||||
|     "hls.js": "1.6.10", |     "hls.js": "1.6.13", | ||||||
|     "home-assistant-js-websocket": "9.5.0", |     "home-assistant-js-websocket": "9.5.0", | ||||||
|     "idb-keyval": "6.2.2", |     "idb-keyval": "6.2.2", | ||||||
|     "intl-messageformat": "10.7.16", |     "intl-messageformat": "10.7.17", | ||||||
|     "js-yaml": "4.1.0", |     "js-yaml": "4.1.0", | ||||||
|     "leaflet": "1.9.4", |     "leaflet": "1.9.4", | ||||||
|     "leaflet-draw": "patch:leaflet-draw@npm%3A1.0.4#./.yarn/patches/leaflet-draw-npm-1.0.4-0ca0ebcf65.patch", |     "leaflet-draw": "patch:leaflet-draw@npm%3A1.0.4#./.yarn/patches/leaflet-draw-npm-1.0.4-0ca0ebcf65.patch", | ||||||
|     "leaflet.markercluster": "1.5.3", |     "leaflet.markercluster": "1.5.3", | ||||||
|     "lit": "3.3.1", |     "lit": "3.3.1", | ||||||
|     "lit-html": "3.3.1", |     "lit-html": "3.3.1", | ||||||
|     "luxon": "3.7.1", |     "luxon": "3.7.2", | ||||||
|     "marked": "16.2.0", |     "marked": "16.4.0", | ||||||
|     "memoize-one": "6.0.0", |     "memoize-one": "6.0.0", | ||||||
|     "node-vibrant": "4.0.3", |     "node-vibrant": "4.0.3", | ||||||
|     "object-hash": "3.0.0", |     "object-hash": "3.0.0", | ||||||
| @@ -136,7 +135,7 @@ | |||||||
|     "stacktrace-js": "2.0.2", |     "stacktrace-js": "2.0.2", | ||||||
|     "superstruct": "2.0.2", |     "superstruct": "2.0.2", | ||||||
|     "tinykeys": "3.0.0", |     "tinykeys": "3.0.0", | ||||||
|     "ua-parser-js": "2.0.4", |     "ua-parser-js": "2.0.5", | ||||||
|     "vue": "2.7.16", |     "vue": "2.7.16", | ||||||
|     "vue2-daterange-picker": "0.6.8", |     "vue2-daterange-picker": "0.6.8", | ||||||
|     "weekstart": "2.0.0", |     "weekstart": "2.0.0", | ||||||
| @@ -149,28 +148,28 @@ | |||||||
|     "xss": "1.0.15" |     "xss": "1.0.15" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@babel/core": "7.28.3", |     "@babel/core": "7.28.4", | ||||||
|     "@babel/helper-define-polyfill-provider": "0.6.5", |     "@babel/helper-define-polyfill-provider": "0.6.5", | ||||||
|     "@babel/plugin-transform-runtime": "7.28.3", |     "@babel/plugin-transform-runtime": "7.28.3", | ||||||
|     "@babel/preset-env": "7.28.3", |     "@babel/preset-env": "7.28.3", | ||||||
|     "@bundle-stats/plugin-webpack-filter": "4.21.3", |     "@bundle-stats/plugin-webpack-filter": "4.21.4", | ||||||
|     "@lokalise/node-api": "15.2.1", |     "@lokalise/node-api": "15.3.0", | ||||||
|     "@octokit/auth-oauth-device": "8.0.1", |     "@octokit/auth-oauth-device": "8.0.2", | ||||||
|     "@octokit/plugin-retry": "8.0.1", |     "@octokit/plugin-retry": "8.0.2", | ||||||
|     "@octokit/rest": "22.0.0", |     "@octokit/rest": "22.0.0", | ||||||
|     "@rsdoctor/rspack-plugin": "1.2.3", |     "@rsdoctor/rspack-plugin": "1.3.1", | ||||||
|     "@rspack/core": "1.4.11", |     "@rspack/core": "1.5.8", | ||||||
|     "@rspack/dev-server": "1.1.4", |     "@rspack/dev-server": "1.1.4", | ||||||
|     "@types/babel__plugin-transform-runtime": "7.9.5", |     "@types/babel__plugin-transform-runtime": "7.9.5", | ||||||
|     "@types/chromecast-caf-receiver": "6.0.22", |     "@types/chromecast-caf-receiver": "6.0.22", | ||||||
|     "@types/chromecast-caf-sender": "1.0.11", |     "@types/chromecast-caf-sender": "1.0.11", | ||||||
|     "@types/color-name": "2.0.0", |     "@types/color-name": "2.0.0", | ||||||
|     "@types/culori": "4.0.0", |     "@types/culori": "4.0.1", | ||||||
|     "@types/html-minifier-terser": "7.0.2", |     "@types/html-minifier-terser": "7.0.2", | ||||||
|     "@types/js-yaml": "4.0.9", |     "@types/js-yaml": "4.0.9", | ||||||
|     "@types/leaflet": "1.9.20", |     "@types/leaflet": "1.9.20", | ||||||
|     "@types/leaflet-draw": "1.0.12", |     "@types/leaflet-draw": "1.0.13", | ||||||
|     "@types/leaflet.markercluster": "1.5.5", |     "@types/leaflet.markercluster": "1.5.6", | ||||||
|     "@types/lodash.merge": "4.6.9", |     "@types/lodash.merge": "4.6.9", | ||||||
|     "@types/luxon": "3.7.1", |     "@types/luxon": "3.7.1", | ||||||
|     "@types/mocha": "10.0.10", |     "@types/mocha": "10.0.10", | ||||||
| @@ -183,8 +182,8 @@ | |||||||
|     "babel-loader": "10.0.0", |     "babel-loader": "10.0.0", | ||||||
|     "babel-plugin-template-html-minifier": "4.1.0", |     "babel-plugin-template-html-minifier": "4.1.0", | ||||||
|     "browserslist-useragent-regexp": "4.1.3", |     "browserslist-useragent-regexp": "4.1.3", | ||||||
|     "del": "8.0.0", |     "del": "8.0.1", | ||||||
|     "eslint": "9.34.0", |     "eslint": "9.37.0", | ||||||
|     "eslint-config-airbnb-base": "15.0.0", |     "eslint-config-airbnb-base": "15.0.0", | ||||||
|     "eslint-config-prettier": "10.1.8", |     "eslint-config-prettier": "10.1.8", | ||||||
|     "eslint-import-resolver-webpack": "0.13.10", |     "eslint-import-resolver-webpack": "0.13.10", | ||||||
| @@ -192,9 +191,9 @@ | |||||||
|     "eslint-plugin-lit": "2.1.1", |     "eslint-plugin-lit": "2.1.1", | ||||||
|     "eslint-plugin-lit-a11y": "5.1.1", |     "eslint-plugin-lit-a11y": "5.1.1", | ||||||
|     "eslint-plugin-unused-imports": "4.2.0", |     "eslint-plugin-unused-imports": "4.2.0", | ||||||
|     "eslint-plugin-wc": "3.0.1", |     "eslint-plugin-wc": "3.0.2", | ||||||
|     "fancy-log": "2.0.0", |     "fancy-log": "2.0.0", | ||||||
|     "fs-extra": "11.3.1", |     "fs-extra": "11.3.2", | ||||||
|     "glob": "11.0.3", |     "glob": "11.0.3", | ||||||
|     "gulp": "5.0.1", |     "gulp": "5.0.1", | ||||||
|     "gulp-brotli": "3.0.0", |     "gulp-brotli": "3.0.0", | ||||||
| @@ -202,23 +201,23 @@ | |||||||
|     "gulp-rename": "2.1.0", |     "gulp-rename": "2.1.0", | ||||||
|     "html-minifier-terser": "7.2.0", |     "html-minifier-terser": "7.2.0", | ||||||
|     "husky": "9.1.7", |     "husky": "9.1.7", | ||||||
|     "jsdom": "26.1.0", |     "jsdom": "27.0.0", | ||||||
|     "jszip": "3.10.1", |     "jszip": "3.10.1", | ||||||
|     "lint-staged": "16.1.5", |     "lint-staged": "16.2.3", | ||||||
|     "lit-analyzer": "2.0.3", |     "lit-analyzer": "2.0.3", | ||||||
|     "lodash.merge": "4.6.2", |     "lodash.merge": "4.6.2", | ||||||
|     "lodash.template": "4.5.0", |     "lodash.template": "4.5.0", | ||||||
|     "map-stream": "0.0.7", |     "map-stream": "0.0.7", | ||||||
|     "pinst": "3.0.0", |     "pinst": "3.0.0", | ||||||
|     "prettier": "3.6.2", |     "prettier": "3.6.2", | ||||||
|     "rspack-manifest-plugin": "5.0.3", |     "rspack-manifest-plugin": "5.1.0", | ||||||
|     "serve": "14.2.4", |     "serve": "14.2.5", | ||||||
|     "sinon": "21.0.0", |     "sinon": "21.0.0", | ||||||
|     "tar": "7.4.3", |     "tar": "7.5.1", | ||||||
|     "terser-webpack-plugin": "5.3.14", |     "terser-webpack-plugin": "5.3.14", | ||||||
|     "ts-lit-plugin": "2.0.2", |     "ts-lit-plugin": "2.0.2", | ||||||
|     "typescript": "5.9.2", |     "typescript": "5.9.3", | ||||||
|     "typescript-eslint": "8.40.0", |     "typescript-eslint": "8.46.0", | ||||||
|     "vite-tsconfig-paths": "5.1.4", |     "vite-tsconfig-paths": "5.1.4", | ||||||
|     "vitest": "3.2.4", |     "vitest": "3.2.4", | ||||||
|     "webpack-stats-plugin": "1.1.3", |     "webpack-stats-plugin": "1.1.3", | ||||||
| @@ -232,10 +231,9 @@ | |||||||
|     "clean-css": "5.3.3", |     "clean-css": "5.3.3", | ||||||
|     "@lit/reactive-element": "2.1.1", |     "@lit/reactive-element": "2.1.1", | ||||||
|     "@fullcalendar/daygrid": "6.1.19", |     "@fullcalendar/daygrid": "6.1.19", | ||||||
|     "globals": "16.3.0", |     "globals": "16.4.0", | ||||||
|     "tslib": "2.8.1", |     "tslib": "2.8.1", | ||||||
|     "@material/mwc-list@^0.27.0": "patch:@material/mwc-list@npm%3A0.27.0#~/.yarn/patches/@material-mwc-list-npm-0.27.0-5344fc9de4.patch", |     "@material/mwc-list@^0.27.0": "patch:@material/mwc-list@npm%3A0.27.0#~/.yarn/patches/@material-mwc-list-npm-0.27.0-5344fc9de4.patch" | ||||||
|     "@vaadin/vaadin-themable-mixin": "24.8.5" |  | ||||||
|   }, |   }, | ||||||
|   "packageManager": "yarn@4.9.3" |   "packageManager": "yarn@4.10.3" | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,3 +1,11 @@ | |||||||
| export default { | export default { | ||||||
|   trailingComma: "es5", |   trailingComma: "es5", | ||||||
|  |   overrides: [ | ||||||
|  |     { | ||||||
|  |       files: "*.globals.ts", | ||||||
|  |       options: { | ||||||
|  |         printWidth: 9999, // Effectively disables line wrapping for these files | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   ], | ||||||
| }; | }; | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 5.4 KiB | 
| @@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta" | |||||||
|  |  | ||||||
| [project] | [project] | ||||||
| name         = "home-assistant-frontend" | name         = "home-assistant-frontend" | ||||||
| version      = "20250903.2" | version      = "20250924.0" | ||||||
| license      = "Apache-2.0" | license      = "Apache-2.0" | ||||||
| license-files = ["LICENSE*"] | license-files = ["LICENSE*"] | ||||||
| description  = "The Home Assistant frontend" | description  = "The Home Assistant frontend" | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ | |||||||
|     ":semanticCommitsDisabled", |     ":semanticCommitsDisabled", | ||||||
|     "group:monorepos", |     "group:monorepos", | ||||||
|     "group:recommended", |     "group:recommended", | ||||||
|     "npm:unpublishSafe" |     "security:minimumReleaseAgeNpm" | ||||||
|   ], |   ], | ||||||
|   "enabledManagers": ["npm", "nvm"], |   "enabledManagers": ["npm", "nvm"], | ||||||
|   "postUpdateOptions": ["yarnDedupeHighest"], |   "postUpdateOptions": ["yarnDedupeHighest"], | ||||||
|   | |||||||
| @@ -103,7 +103,10 @@ export class HaAuthorize extends litLocalizeLiteMixin(LitElement) { | |||||||
|           ); |           ); | ||||||
|           box-shadow: var(--ha-card-box-shadow, none); |           box-shadow: var(--ha-card-box-shadow, none); | ||||||
|           box-sizing: border-box; |           box-sizing: border-box; | ||||||
|           border-radius: var(--ha-card-border-radius, 12px); |           border-radius: var( | ||||||
|  |             --ha-card-border-radius, | ||||||
|  |             var(--ha-border-radius-lg) | ||||||
|  |           ); | ||||||
|           border-width: var(--ha-card-border-width, 1px); |           border-width: var(--ha-card-border-width, 1px); | ||||||
|           border-style: solid; |           border-style: solid; | ||||||
|           border-color: var( |           border-color: var( | ||||||
| @@ -132,7 +135,7 @@ export class HaAuthorize extends litLocalizeLiteMixin(LitElement) { | |||||||
|         } |         } | ||||||
|         ha-language-picker { |         ha-language-picker { | ||||||
|           width: 200px; |           width: 200px; | ||||||
|           border-radius: 4px; |           border-radius: var(--ha-border-radius-sm); | ||||||
|           overflow: hidden; |           overflow: hidden; | ||||||
|           --ha-select-height: 40px; |           --ha-select-height: 40px; | ||||||
|           --mdc-select-fill-color: none; |           --mdc-select-fill-color: none; | ||||||
|   | |||||||
| @@ -1,23 +1,40 @@ | |||||||
|  | import { formatHex, parse } from "culori"; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Expands a 3-digit hex color to a 6-digit hex color. | ||||||
|  |  * @param hex - The hex color to expand. | ||||||
|  |  * @returns The expanded hex color. | ||||||
|  |  * @throws If the hex color is invalid. | ||||||
|  |  */ | ||||||
| export const expandHex = (hex: string): string => { | export const expandHex = (hex: string): string => { | ||||||
|   hex = hex.replace("#", ""); |   const color = parse(hex); | ||||||
|   if (hex.length === 6) return hex; |   if (!color) { | ||||||
|   let result = ""; |     throw new Error(`Invalid hex color: ${hex}`); | ||||||
|   for (const val of hex) { |  | ||||||
|     result += val + val; |  | ||||||
|   } |   } | ||||||
|   return result; |   const formattedColor = formatHex(color); | ||||||
|  |   if (!formattedColor) { | ||||||
|  |     throw new Error(`Could not format hex color: ${hex}`); | ||||||
|  |   } | ||||||
|  |   return formattedColor.replace("#", ""); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| // Blend 2 hex colors: c1 is placed over c2, blend is c1's opacity. | /** | ||||||
|  |  * Blends two hex colors. c1 is placed over c2, blend is c1's opacity. | ||||||
|  |  * @param c1 - The first hex color. | ||||||
|  |  * @param c2 - The second hex color. | ||||||
|  |  * @param blend - The blend percentage (0-100). | ||||||
|  |  * @returns The blended hex color. | ||||||
|  |  */ | ||||||
| export const hexBlend = (c1: string, c2: string, blend = 50): string => { | export const hexBlend = (c1: string, c2: string, blend = 50): string => { | ||||||
|   let color = ""; |  | ||||||
|   c1 = expandHex(c1); |   c1 = expandHex(c1); | ||||||
|   c2 = expandHex(c2); |   c2 = expandHex(c2); | ||||||
|  |   let color = ""; | ||||||
|   for (let i = 0; i <= 5; i += 2) { |   for (let i = 0; i <= 5; i += 2) { | ||||||
|     const h1 = parseInt(c1.substring(i, i + 2), 16); |     const h1 = parseInt(c1.substring(i, i + 2), 16); | ||||||
|     const h2 = parseInt(c2.substring(i, i + 2), 16); |     const h2 = parseInt(c2.substring(i, i + 2), 16); | ||||||
|     let hex = Math.floor(h2 + (h1 - h2) * (blend / 100)).toString(16); |     const hex = Math.floor(h2 + (h1 - h2) * (blend / 100)) | ||||||
|     while (hex.length < 2) hex = "0" + hex; |       .toString(16) | ||||||
|  |       .padStart(2, "0"); | ||||||
|     color += hex; |     color += hex; | ||||||
|   } |   } | ||||||
|   return `#${color}`; |   return `#${color}`; | ||||||
|   | |||||||
| @@ -1,28 +1,49 @@ | |||||||
| export const luminosity = (rgb: [number, number, number]): number => { | import { wcagLuminance, wcagContrast } from "culori"; | ||||||
|   // http://www.w3.org/TR/WCAG20/#relativeluminancedef |  | ||||||
|   const lum: [number, number, number] = [0, 0, 0]; |  | ||||||
|   for (let i = 0; i < rgb.length; i++) { |  | ||||||
|     const chan = rgb[i] / 255; |  | ||||||
|     lum[i] = chan <= 0.03928 ? chan / 12.92 : ((chan + 0.055) / 1.055) ** 2.4; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return 0.2126 * lum[0] + 0.7152 * lum[1] + 0.0722 * lum[2]; | /** | ||||||
| }; |  * Calculates the luminosity of an RGB color. | ||||||
|  |  * @param rgb - The RGB color to calculate the luminosity of. | ||||||
|  |  * @returns The luminosity of the color. | ||||||
|  |  */ | ||||||
|  | export const luminosity = (rgb: [number, number, number]): number => | ||||||
|  |   wcagLuminance({ | ||||||
|  |     mode: "rgb", | ||||||
|  |     r: rgb[0] / 255, | ||||||
|  |     g: rgb[1] / 255, | ||||||
|  |     b: rgb[2] / 255, | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Calculates the contrast ratio between two RGB colors. | ||||||
|  |  * @param color1 - The first color to calculate the contrast ratio of. | ||||||
|  |  * @param color2 - The second color to calculate the contrast ratio of. | ||||||
|  |  * @returns The contrast ratio between the two colors. | ||||||
|  |  */ | ||||||
| export const rgbContrast = ( | export const rgbContrast = ( | ||||||
|   color1: [number, number, number], |   color1: [number, number, number], | ||||||
|   color2: [number, number, number] |   color2: [number, number, number] | ||||||
| ) => { | ) => | ||||||
|   const lum1 = luminosity(color1); |   wcagContrast( | ||||||
|   const lum2 = luminosity(color2); |     { | ||||||
|  |       mode: "rgb", | ||||||
|   if (lum1 > lum2) { |       r: color1[0] / 255, | ||||||
|     return (lum1 + 0.05) / (lum2 + 0.05); |       g: color1[1] / 255, | ||||||
|  |       b: color1[2] / 255, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       mode: "rgb", | ||||||
|  |       r: color2[0] / 255, | ||||||
|  |       g: color2[1] / 255, | ||||||
|  |       b: color2[2] / 255, | ||||||
|     } |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|   return (lum2 + 0.05) / (lum1 + 0.05); | /** | ||||||
| }; |  * Calculates the contrast ratio between two RGB colors. | ||||||
|  |  * @param rgb1 - The first color to calculate the contrast ratio of. | ||||||
|  |  * @param rgb2 - The second color to calculate the contrast ratio of. | ||||||
|  |  * @returns The contrast ratio between the two colors. | ||||||
|  |  */ | ||||||
| export const getRGBContrastRatio = ( | export const getRGBContrastRatio = ( | ||||||
|   rgb1: [number, number, number], |   rgb1: [number, number, number], | ||||||
|   rgb2: [number, number, number] |   rgb2: [number, number, number] | ||||||
|   | |||||||
							
								
								
									
										141
									
								
								src/common/controllers/undo-redo-controller.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								src/common/controllers/undo-redo-controller.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,141 @@ | |||||||
|  | import type { | ||||||
|  |   ReactiveController, | ||||||
|  |   ReactiveControllerHost, | ||||||
|  | } from "@lit/reactive-element/reactive-controller"; | ||||||
|  |  | ||||||
|  | const UNDO_REDO_STACK_LIMIT = 75; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Configuration options for the UndoRedoController. | ||||||
|  |  * | ||||||
|  |  * @template ConfigType The type of configuration to manage. | ||||||
|  |  */ | ||||||
|  | export interface UndoRedoControllerConfig<ConfigType> { | ||||||
|  |   stackLimit?: number; | ||||||
|  |   currentConfig: () => ConfigType; | ||||||
|  |   apply: (config: ConfigType) => void; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * A controller to manage undo and redo operations for a given configuration type. | ||||||
|  |  * | ||||||
|  |  * @template ConfigType The type of configuration to manage. | ||||||
|  |  */ | ||||||
|  | export class UndoRedoController<ConfigType> implements ReactiveController { | ||||||
|  |   private _host: ReactiveControllerHost; | ||||||
|  |  | ||||||
|  |   private _undoStack: ConfigType[] = []; | ||||||
|  |  | ||||||
|  |   private _redoStack: ConfigType[] = []; | ||||||
|  |  | ||||||
|  |   private readonly _stackLimit: number = UNDO_REDO_STACK_LIMIT; | ||||||
|  |  | ||||||
|  |   private readonly _apply: (config: ConfigType) => void = () => { | ||||||
|  |     throw new Error("No apply function provided"); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   private readonly _currentConfig: () => ConfigType = () => { | ||||||
|  |     throw new Error("No currentConfig function provided"); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   constructor( | ||||||
|  |     host: ReactiveControllerHost, | ||||||
|  |     options: UndoRedoControllerConfig<ConfigType> | ||||||
|  |   ) { | ||||||
|  |     if (options.stackLimit !== undefined) { | ||||||
|  |       this._stackLimit = options.stackLimit; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     this._apply = options.apply; | ||||||
|  |     this._currentConfig = options.currentConfig; | ||||||
|  |     this._host = host; | ||||||
|  |     host.addController(this); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   hostConnected() { | ||||||
|  |     window.addEventListener("undo-change", this._onUndoChange); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   hostDisconnected() { | ||||||
|  |     window.removeEventListener("undo-change", this._onUndoChange); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _onUndoChange = (ev: Event) => { | ||||||
|  |     ev.stopPropagation(); | ||||||
|  |     this.undo(); | ||||||
|  |     this._host.requestUpdate(); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Indicates whether there are actions available to undo. | ||||||
|  |    * | ||||||
|  |    * @returns `true` if there are actions to undo, `false` otherwise. | ||||||
|  |    */ | ||||||
|  |   public get canUndo(): boolean { | ||||||
|  |     return this._undoStack.length > 0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Indicates whether there are actions available to redo. | ||||||
|  |    * | ||||||
|  |    * @returns `true` if there are actions to redo, `false` otherwise. | ||||||
|  |    */ | ||||||
|  |   public get canRedo(): boolean { | ||||||
|  |     return this._redoStack.length > 0; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Commits the current configuration to the undo stack and clears the redo stack. | ||||||
|  |    * | ||||||
|  |    * @param config The current configuration to commit. | ||||||
|  |    */ | ||||||
|  |   public commit(config: ConfigType) { | ||||||
|  |     if (this._undoStack.length >= this._stackLimit) { | ||||||
|  |       this._undoStack.shift(); | ||||||
|  |     } | ||||||
|  |     this._undoStack.push({ ...config }); | ||||||
|  |     this._redoStack = []; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Undoes the last action and applies the previous configuration | ||||||
|  |    * while saving the current configuration to the redo stack. | ||||||
|  |    */ | ||||||
|  |   public undo() { | ||||||
|  |     if (this._undoStack.length === 0) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     this._redoStack.push({ ...this._currentConfig() }); | ||||||
|  |     const config = this._undoStack.pop()!; | ||||||
|  |     this._apply(config); | ||||||
|  |     this._host.requestUpdate(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Redoes the last undone action and reapplies the configuration | ||||||
|  |    * while saving the current configuration to the undo stack. | ||||||
|  |    */ | ||||||
|  |   public redo() { | ||||||
|  |     if (this._redoStack.length === 0) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     this._undoStack.push({ ...this._currentConfig() }); | ||||||
|  |     const config = this._redoStack.pop()!; | ||||||
|  |     this._apply(config); | ||||||
|  |     this._host.requestUpdate(); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Resets the undo and redo stacks, clearing all history. | ||||||
|  |    */ | ||||||
|  |   public reset() { | ||||||
|  |     this._undoStack = []; | ||||||
|  |     this._redoStack = []; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare global { | ||||||
|  |   interface HASSDomEvents { | ||||||
|  |     "undo-change": undefined; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -11,7 +11,7 @@ import { | |||||||
|   differenceInDays, |   differenceInDays, | ||||||
|   addDays, |   addDays, | ||||||
| } from "date-fns"; | } from "date-fns"; | ||||||
| import { toZonedTime, fromZonedTime } from "date-fns-tz"; | import { TZDate } from "@date-fns/tz"; | ||||||
| import type { HassConfig } from "home-assistant-js-websocket"; | import type { HassConfig } from "home-assistant-js-websocket"; | ||||||
| import type { FrontendLocaleData } from "../../data/translation"; | import type { FrontendLocaleData } from "../../data/translation"; | ||||||
| import { TimeZone } from "../../data/translation"; | import { TimeZone } from "../../data/translation"; | ||||||
| @@ -22,12 +22,13 @@ const calcZonedDate = ( | |||||||
|   fn: (date: Date, options?: any) => Date | number | boolean, |   fn: (date: Date, options?: any) => Date | number | boolean, | ||||||
|   options? |   options? | ||||||
| ) => { | ) => { | ||||||
|   const inputZoned = toZonedTime(date, tz); |   const tzDate = new TZDate(date, tz); | ||||||
|   const fnZoned = fn(inputZoned, options); |   const fnResult = fn(tzDate, options); | ||||||
|   if (fnZoned instanceof Date) { |   if (fnResult instanceof Date) { | ||||||
|     return fromZonedTime(fnZoned, tz) as Date; |     // Convert back to regular Date in the specified timezone | ||||||
|  |     return new Date(fnResult.getTime()); | ||||||
|   } |   } | ||||||
|   return fnZoned; |   return fnResult; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const calcDate = ( | export const calcDate = ( | ||||||
| @@ -65,7 +66,7 @@ export const calcDateDifferenceProperty = ( | |||||||
|     locale, |     locale, | ||||||
|     config, |     config, | ||||||
|     locale.time_zone === TimeZone.server |     locale.time_zone === TimeZone.server | ||||||
|       ? toZonedTime(startDate, config.time_zone) |       ? new TZDate(startDate, config.time_zone) | ||||||
|       : startDate |       : startDate | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
| @@ -144,3 +145,36 @@ export const shiftDateRange = ( | |||||||
|   } |   } | ||||||
|   return { start, end }; |   return { start, end }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @description Parses a date in browser display timezone | ||||||
|  |  * @param date - The date to parse | ||||||
|  |  * @param timezone - The timezone to parse the date in | ||||||
|  |  * @returns The parsed date as a Date object | ||||||
|  |  */ | ||||||
|  | export const parseDate = (date: string, timezone: string): Date => { | ||||||
|  |   const tzDate = new TZDate(date, timezone); | ||||||
|  |   return new Date(tzDate.getTime()); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @description Formats a date in browser display timezone | ||||||
|  |  * @param date - The date to format | ||||||
|  |  * @param timezone - The timezone to format the date in | ||||||
|  |  * @returns The formatted date in YYYY-MM-DD format | ||||||
|  |  */ | ||||||
|  | export const formatDate = (date: Date, timezone: string): string => { | ||||||
|  |   const tzDate = new TZDate(date, timezone); | ||||||
|  |   return tzDate.toISOString().split("T")[0]; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @description Formats a time in browser display timezone | ||||||
|  |  * @param date - The date to format | ||||||
|  |  * @param timezone - The timezone to format the time in | ||||||
|  |  * @returns The formatted time in HH:mm:ss format | ||||||
|  |  */ | ||||||
|  | export const formatTime = (date: Date, timezone: string): string => { | ||||||
|  |   const tzDate = new TZDate(date, timezone); | ||||||
|  |   return tzDate.toISOString().split("T")[1].split(".")[0]; | ||||||
|  | }; | ||||||
|   | |||||||
| @@ -31,10 +31,10 @@ export const isNavigationClick = (e: MouseEvent, preventDefault = true) => { | |||||||
|  |  | ||||||
|   const location = window.location; |   const location = window.location; | ||||||
|   const origin = location.origin || location.protocol + "//" + location.host; |   const origin = location.origin || location.protocol + "//" + location.host; | ||||||
|   if (href.indexOf(origin) !== 0) { |   if (!href.startsWith(origin)) { | ||||||
|     return undefined; |     return undefined; | ||||||
|   } |   } | ||||||
|   href = href.substr(origin.length); |   href = href.slice(origin.length); | ||||||
|  |  | ||||||
|   if (href === "#") { |   if (href === "#") { | ||||||
|     return undefined; |     return undefined; | ||||||
|   | |||||||
| @@ -10,9 +10,10 @@ import { stripPrefixFromEntityName } from "./strip_prefix_from_entity_name"; | |||||||
|  |  | ||||||
| export const computeEntityName = ( | export const computeEntityName = ( | ||||||
|   stateObj: HassEntity, |   stateObj: HassEntity, | ||||||
|   hass: HomeAssistant |   entities: HomeAssistant["entities"], | ||||||
|  |   devices: HomeAssistant["devices"] | ||||||
| ): string | undefined => { | ): string | undefined => { | ||||||
|   const entry = hass.entities[stateObj.entity_id] as |   const entry = entities[stateObj.entity_id] as | ||||||
|     | EntityRegistryDisplayEntry |     | EntityRegistryDisplayEntry | ||||||
|     | undefined; |     | undefined; | ||||||
|  |  | ||||||
| @@ -20,12 +21,13 @@ export const computeEntityName = ( | |||||||
|     // Fall back to state name if not in the entity registry (friendly name) |     // Fall back to state name if not in the entity registry (friendly name) | ||||||
|     return computeStateName(stateObj); |     return computeStateName(stateObj); | ||||||
|   } |   } | ||||||
|   return computeEntityEntryName(entry, hass); |   return computeEntityEntryName(entry, devices); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const computeEntityEntryName = ( | export const computeEntityEntryName = ( | ||||||
|   entry: EntityRegistryDisplayEntry | EntityRegistryEntry, |   entry: EntityRegistryDisplayEntry | EntityRegistryEntry, | ||||||
|   hass: HomeAssistant |   devices: HomeAssistant["devices"], | ||||||
|  |   fallbackStateObj?: HassEntity | ||||||
| ): string | undefined => { | ): string | undefined => { | ||||||
|   const name = |   const name = | ||||||
|     entry.name || |     entry.name || | ||||||
| @@ -33,15 +35,14 @@ export const computeEntityEntryName = ( | |||||||
|       ? String(entry.original_name) |       ? String(entry.original_name) | ||||||
|       : undefined); |       : undefined); | ||||||
|  |  | ||||||
|   const device = entry.device_id ? hass.devices[entry.device_id] : undefined; |   const device = entry.device_id ? devices[entry.device_id] : undefined; | ||||||
|  |  | ||||||
|   if (!device) { |   if (!device) { | ||||||
|     if (name) { |     if (name) { | ||||||
|       return name; |       return name; | ||||||
|     } |     } | ||||||
|     const stateObj = hass.states[entry.entity_id] as HassEntity | undefined; |     if (fallbackStateObj) { | ||||||
|     if (stateObj) { |       return computeStateName(fallbackStateObj); | ||||||
|       return computeStateName(stateObj); |  | ||||||
|     } |     } | ||||||
|     return undefined; |     return undefined; | ||||||
|   } |   } | ||||||
| @@ -60,3 +61,9 @@ export const computeEntityEntryName = ( | |||||||
|  |  | ||||||
|   return name; |   return name; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | export const entityUseDeviceName = ( | ||||||
|  |   stateObj: HassEntity, | ||||||
|  |   entities: HomeAssistant["entities"], | ||||||
|  |   devices: HomeAssistant["devices"] | ||||||
|  | ): boolean => !computeEntityName(stateObj, entities, devices); | ||||||
|   | |||||||
							
								
								
									
										104
									
								
								src/common/entity/compute_entity_name_display.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								src/common/entity/compute_entity_name_display.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,104 @@ | |||||||
|  | import type { HassEntity } from "home-assistant-js-websocket"; | ||||||
|  | import type { HomeAssistant } from "../../types"; | ||||||
|  | import { ensureArray } from "../array/ensure-array"; | ||||||
|  | import { computeAreaName } from "./compute_area_name"; | ||||||
|  | import { computeDeviceName } from "./compute_device_name"; | ||||||
|  | import { computeEntityName, entityUseDeviceName } from "./compute_entity_name"; | ||||||
|  | import { computeFloorName } from "./compute_floor_name"; | ||||||
|  | import { getEntityContext } from "./context/get_entity_context"; | ||||||
|  |  | ||||||
|  | const DEFAULT_SEPARATOR = " "; | ||||||
|  |  | ||||||
|  | export type EntityNameItem = | ||||||
|  |   | { | ||||||
|  |       type: "entity" | "device" | "area" | "floor"; | ||||||
|  |     } | ||||||
|  |   | { | ||||||
|  |       type: "text"; | ||||||
|  |       text: string; | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  | export interface EntityNameOptions { | ||||||
|  |   separator?: string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export const computeEntityNameDisplay = ( | ||||||
|  |   stateObj: HassEntity, | ||||||
|  |   name: EntityNameItem | EntityNameItem[], | ||||||
|  |   entities: HomeAssistant["entities"], | ||||||
|  |   devices: HomeAssistant["devices"], | ||||||
|  |   areas: HomeAssistant["areas"], | ||||||
|  |   floors: HomeAssistant["floors"], | ||||||
|  |   options?: EntityNameOptions | ||||||
|  | ) => { | ||||||
|  |   let items = ensureArray(name); | ||||||
|  |  | ||||||
|  |   const separator = options?.separator ?? DEFAULT_SEPARATOR; | ||||||
|  |  | ||||||
|  |   // If all items are text, just join them | ||||||
|  |   if (items.every((n) => n.type === "text")) { | ||||||
|  |     return items.map((item) => item.text).join(separator); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const useDeviceName = entityUseDeviceName(stateObj, entities, devices); | ||||||
|  |  | ||||||
|  |   // If entity uses device name, and device is not already included, replace it with device name | ||||||
|  |   if (useDeviceName) { | ||||||
|  |     const hasDevice = items.some((n) => n.type === "device"); | ||||||
|  |     if (!hasDevice) { | ||||||
|  |       items = items.map((n) => (n.type === "entity" ? { type: "device" } : n)); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const names = computeEntityNameList( | ||||||
|  |     stateObj, | ||||||
|  |     items, | ||||||
|  |     entities, | ||||||
|  |     devices, | ||||||
|  |     areas, | ||||||
|  |     floors | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   // If after processing there is only one name, return that | ||||||
|  |   if (names.length === 1) { | ||||||
|  |     return names[0] || ""; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return names.filter((n) => n).join(separator); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export const computeEntityNameList = ( | ||||||
|  |   stateObj: HassEntity, | ||||||
|  |   name: EntityNameItem[], | ||||||
|  |   entities: HomeAssistant["entities"], | ||||||
|  |   devices: HomeAssistant["devices"], | ||||||
|  |   areas: HomeAssistant["areas"], | ||||||
|  |   floors: HomeAssistant["floors"] | ||||||
|  | ): (string | undefined)[] => { | ||||||
|  |   const { device, area, floor } = getEntityContext( | ||||||
|  |     stateObj, | ||||||
|  |     entities, | ||||||
|  |     devices, | ||||||
|  |     areas, | ||||||
|  |     floors | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   const names = name.map((item) => { | ||||||
|  |     switch (item.type) { | ||||||
|  |       case "entity": | ||||||
|  |         return computeEntityName(stateObj, entities, devices); | ||||||
|  |       case "device": | ||||||
|  |         return device ? computeDeviceName(device) : undefined; | ||||||
|  |       case "area": | ||||||
|  |         return area ? computeAreaName(area) : undefined; | ||||||
|  |       case "floor": | ||||||
|  |         return floor ? computeFloorName(floor) : undefined; | ||||||
|  |       case "text": | ||||||
|  |         return item.text; | ||||||
|  |       default: | ||||||
|  |         return ""; | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   return names; | ||||||
|  | }; | ||||||
| @@ -1,3 +1,3 @@ | |||||||
| /** Compute the object ID of a state. */ | /** Compute the object ID of a state. */ | ||||||
| export const computeObjectId = (entityId: string): string => | export const computeObjectId = (entityId: string): string => | ||||||
|   entityId.substr(entityId.indexOf(".") + 1); |   entityId.slice(entityId.indexOf(".") + 1); | ||||||
|   | |||||||
| @@ -18,9 +18,12 @@ interface EntityContext { | |||||||
|  |  | ||||||
| export const getEntityContext = ( | export const getEntityContext = ( | ||||||
|   stateObj: HassEntity, |   stateObj: HassEntity, | ||||||
|   hass: HomeAssistant |   entities: HomeAssistant["entities"], | ||||||
|  |   devices: HomeAssistant["devices"], | ||||||
|  |   areas: HomeAssistant["areas"], | ||||||
|  |   floors: HomeAssistant["floors"] | ||||||
| ): EntityContext => { | ): EntityContext => { | ||||||
|   const entry = hass.entities[stateObj.entity_id] as |   const entry = entities[stateObj.entity_id] as | ||||||
|     | EntityRegistryDisplayEntry |     | EntityRegistryDisplayEntry | ||||||
|     | undefined; |     | undefined; | ||||||
|  |  | ||||||
| @@ -32,7 +35,7 @@ export const getEntityContext = ( | |||||||
|       floor: null, |       floor: null, | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
|   return getEntityEntryContext(entry, hass); |   return getEntityEntryContext(entry, entities, devices, areas, floors); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export const getEntityEntryContext = ( | export const getEntityEntryContext = ( | ||||||
| @@ -40,15 +43,18 @@ export const getEntityEntryContext = ( | |||||||
|     | EntityRegistryDisplayEntry |     | EntityRegistryDisplayEntry | ||||||
|     | EntityRegistryEntry |     | EntityRegistryEntry | ||||||
|     | ExtEntityRegistryEntry, |     | ExtEntityRegistryEntry, | ||||||
|   hass: HomeAssistant |   entities: HomeAssistant["entities"], | ||||||
|  |   devices: HomeAssistant["devices"], | ||||||
|  |   areas: HomeAssistant["areas"], | ||||||
|  |   floors: HomeAssistant["floors"] | ||||||
| ): EntityContext => { | ): EntityContext => { | ||||||
|   const entity = hass.entities[entry.entity_id]; |   const entity = entities[entry.entity_id]; | ||||||
|   const deviceId = entry?.device_id; |   const deviceId = entry?.device_id; | ||||||
|   const device = deviceId ? hass.devices[deviceId] : undefined; |   const device = deviceId ? devices[deviceId] : undefined; | ||||||
|   const areaId = entry?.area_id || device?.area_id; |   const areaId = entry?.area_id || device?.area_id; | ||||||
|   const area = areaId ? hass.areas[areaId] : undefined; |   const area = areaId ? areas[areaId] : undefined; | ||||||
|   const floorId = area?.floor_id; |   const floorId = area?.floor_id; | ||||||
|   const floor = floorId ? hass.floors[floorId] : undefined; |   const floor = floorId ? floors[floorId] : undefined; | ||||||
|  |  | ||||||
|   return { |   return { | ||||||
|     entity: entity, |     entity: entity, | ||||||
|   | |||||||
| @@ -60,17 +60,20 @@ export const generateEntityFilter = ( | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const { area, floor, device, entity } = getEntityContext(stateObj, hass); |     const { area, floor, device, entity } = getEntityContext( | ||||||
|  |       stateObj, | ||||||
|  |       hass.entities, | ||||||
|  |       hass.devices, | ||||||
|  |       hass.areas, | ||||||
|  |       hass.floors | ||||||
|  |     ); | ||||||
|  |  | ||||||
|     if (entity && entity.hidden) { |     if (entity && entity.hidden) { | ||||||
|       return false; |       return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (floors) { |     if (floors) { | ||||||
|       if (!floor) { |       if (!floor || !floors.has(floor.floor_id)) { | ||||||
|         return false; |  | ||||||
|       } |  | ||||||
|       if (!floors) { |  | ||||||
|         return false; |         return false; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
| @@ -119,3 +122,22 @@ export const generateEntityFilter = ( | |||||||
|     return true; |     return true; | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | export const findEntities = ( | ||||||
|  |   entities: string[], | ||||||
|  |   filters: EntityFilterFunc[] | ||||||
|  | ): string[] => { | ||||||
|  |   const seen = new Set<string>(); | ||||||
|  |   const results: string[] = []; | ||||||
|  |  | ||||||
|  |   for (const filter of filters) { | ||||||
|  |     for (const entity of entities) { | ||||||
|  |       if (filter(entity) && !seen.has(entity)) { | ||||||
|  |         seen.add(entity); | ||||||
|  |         results.push(entity); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return results; | ||||||
|  | }; | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ export const FIXED_DOMAIN_STATES = { | |||||||
|     "pending", |     "pending", | ||||||
|     "triggered", |     "triggered", | ||||||
|   ], |   ], | ||||||
|  |   alert: ["on", "off", "idle"], | ||||||
|   assist_satellite: ["idle", "listening", "responding", "processing"], |   assist_satellite: ["idle", "listening", "responding", "processing"], | ||||||
|   automation: ["on", "off"], |   automation: ["on", "off"], | ||||||
|   binary_sensor: ["on", "off"], |   binary_sensor: ["on", "off"], | ||||||
|   | |||||||
| @@ -40,6 +40,7 @@ const STATE_COLORED_DOMAIN = new Set([ | |||||||
|   "vacuum", |   "vacuum", | ||||||
|   "valve", |   "valve", | ||||||
|   "water_heater", |   "water_heater", | ||||||
|  |   "weather", | ||||||
| ]); | ]); | ||||||
|  |  | ||||||
| export const stateColorCss = (stateObj: HassEntity, state?: string) => { | export const stateColorCss = (stateObj: HassEntity, state?: string) => { | ||||||
|   | |||||||
| @@ -63,3 +63,21 @@ export const navigate = async ( | |||||||
|   }); |   }); | ||||||
|   return true; |   return true; | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Navigate back in history, with fallback to a default path if no history exists. | ||||||
|  |  * This prevents a user from getting stuck when they navigate directly to a page with no history. | ||||||
|  |  */ | ||||||
|  | export const goBack = (fallbackPath?: string) => { | ||||||
|  |   const { history } = mainWindow; | ||||||
|  |  | ||||||
|  |   // Check if we have history to go back to | ||||||
|  |   if (history.length > 1) { | ||||||
|  |     history.back(); | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // No history available, navigate to fallback path | ||||||
|  |   const fallback = fallbackPath || "/"; | ||||||
|  |   navigate(fallback, { replace: true }); | ||||||
|  | }; | ||||||
|   | |||||||
| @@ -32,6 +32,8 @@ export const numberFormatToLocale = ( | |||||||
|       return ["de", "es", "it"]; // Use German with fallback to Spanish then Italian formatting 1.234.567,89 |       return ["de", "es", "it"]; // Use German with fallback to Spanish then Italian formatting 1.234.567,89 | ||||||
|     case NumberFormat.space_comma: |     case NumberFormat.space_comma: | ||||||
|       return ["fr", "sv", "cs"]; // Use French with fallback to Swedish and Czech formatting 1 234 567,89 |       return ["fr", "sv", "cs"]; // Use French with fallback to Swedish and Czech formatting 1 234 567,89 | ||||||
|  |     case NumberFormat.quote_decimal: | ||||||
|  |       return ["de-CH"]; // Use German (Switzerland) formatting 1'234'567.89 | ||||||
|     case NumberFormat.system: |     case NumberFormat.system: | ||||||
|       return undefined; |       return undefined; | ||||||
|     default: |     default: | ||||||
|   | |||||||
| @@ -67,10 +67,7 @@ function isSeparatorAtPos(value: string, index: number): boolean { | |||||||
|     case undefined: |     case undefined: | ||||||
|       return false; |       return false; | ||||||
|     default: |     default: | ||||||
|       if (isEmojiImprecise(code)) { |       return isEmojiImprecise(code); | ||||||
|         return true; |  | ||||||
|       } |  | ||||||
|       return false; |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,11 @@ | |||||||
| import type { HassConfig, HassEntity } from "home-assistant-js-websocket"; | import type { HassConfig, HassEntity } from "home-assistant-js-websocket"; | ||||||
| import type { FrontendLocaleData } from "../../data/translation"; | import type { FrontendLocaleData } from "../../data/translation"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
|  | import { | ||||||
|  |   computeEntityNameDisplay, | ||||||
|  |   type EntityNameItem, | ||||||
|  |   type EntityNameOptions, | ||||||
|  | } from "../entity/compute_entity_name_display"; | ||||||
| import type { LocalizeFunc } from "./localize"; | import type { LocalizeFunc } from "./localize"; | ||||||
|  |  | ||||||
| export type FormatEntityStateFunc = ( | export type FormatEntityStateFunc = ( | ||||||
| @@ -17,16 +22,28 @@ export type FormatEntityAttributeNameFunc = ( | |||||||
|   attribute: string |   attribute: string | ||||||
| ) => string; | ) => string; | ||||||
|  |  | ||||||
|  | export type EntityNameType = "entity" | "device" | "area" | "floor"; | ||||||
|  |  | ||||||
|  | export type FormatEntityNameFunc = ( | ||||||
|  |   stateObj: HassEntity, | ||||||
|  |   name: EntityNameItem | EntityNameItem[], | ||||||
|  |   options?: EntityNameOptions | ||||||
|  | ) => string; | ||||||
|  |  | ||||||
| export const computeFormatFunctions = async ( | export const computeFormatFunctions = async ( | ||||||
|   localize: LocalizeFunc, |   localize: LocalizeFunc, | ||||||
|   locale: FrontendLocaleData, |   locale: FrontendLocaleData, | ||||||
|   config: HassConfig, |   config: HassConfig, | ||||||
|   entities: HomeAssistant["entities"], |   entities: HomeAssistant["entities"], | ||||||
|  |   devices: HomeAssistant["devices"], | ||||||
|  |   areas: HomeAssistant["areas"], | ||||||
|  |   floors: HomeAssistant["floors"], | ||||||
|   sensorNumericDeviceClasses: string[] |   sensorNumericDeviceClasses: string[] | ||||||
| ): Promise<{ | ): Promise<{ | ||||||
|   formatEntityState: FormatEntityStateFunc; |   formatEntityState: FormatEntityStateFunc; | ||||||
|   formatEntityAttributeValue: FormatEntityAttributeValueFunc; |   formatEntityAttributeValue: FormatEntityAttributeValueFunc; | ||||||
|   formatEntityAttributeName: FormatEntityAttributeNameFunc; |   formatEntityAttributeName: FormatEntityAttributeNameFunc; | ||||||
|  |   formatEntityName: FormatEntityNameFunc; | ||||||
| }> => { | }> => { | ||||||
|   const { computeStateDisplay } = await import( |   const { computeStateDisplay } = await import( | ||||||
|     "../entity/compute_state_display" |     "../entity/compute_state_display" | ||||||
| @@ -57,5 +74,15 @@ export const computeFormatFunctions = async ( | |||||||
|       ), |       ), | ||||||
|     formatEntityAttributeName: (stateObj, attribute) => |     formatEntityAttributeName: (stateObj, attribute) => | ||||||
|       computeAttributeNameDisplay(localize, stateObj, entities, attribute), |       computeAttributeNameDisplay(localize, stateObj, entities, attribute), | ||||||
|  |     formatEntityName: (stateObj, name, options) => | ||||||
|  |       computeEntityNameDisplay( | ||||||
|  |         stateObj, | ||||||
|  |         name, | ||||||
|  |         entities, | ||||||
|  |         devices, | ||||||
|  |         areas, | ||||||
|  |         floors, | ||||||
|  |         options | ||||||
|  |       ), | ||||||
|   }; |   }; | ||||||
| }; | }; | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								src/common/util/order-properties.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/common/util/order-properties.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | /** | ||||||
|  |  * Orders object properties according to a specified key order. | ||||||
|  |  * Properties not in the order array will be placed at the end. | ||||||
|  |  */ | ||||||
|  | export function orderProperties<T extends Record<string, any>>( | ||||||
|  |   obj: T, | ||||||
|  |   keys: readonly string[] | ||||||
|  | ): T { | ||||||
|  |   const orderedEntries = keys | ||||||
|  |     .filter((key) => key in obj) | ||||||
|  |     .map((key) => [key, obj[key]] as const); | ||||||
|  |  | ||||||
|  |   const extraEntries = Object.entries(obj).filter( | ||||||
|  |     ([key]) => !keys.includes(key) | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   return Object.fromEntries([...orderedEntries, ...extraEntries]) as T; | ||||||
|  | } | ||||||
| @@ -14,7 +14,7 @@ export default function parseAspectRatio(input: string) { | |||||||
|   } |   } | ||||||
|   try { |   try { | ||||||
|     if (input.endsWith("%")) { |     if (input.endsWith("%")) { | ||||||
|       return { w: 100, h: parseOrThrow(input.substr(0, input.length - 1)) }; |       return { w: 100, h: parseOrThrow(input.slice(0, -1)) }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const arr = input.replace(":", "x").split("x"); |     const arr = input.replace(":", "x").split("x"); | ||||||
|   | |||||||
							
								
								
									
										8
									
								
								src/common/util/xss.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								src/common/util/xss.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | import xss from "xss"; | ||||||
|  |  | ||||||
|  | export const filterXSS = (html: string) => | ||||||
|  |   xss(html, { | ||||||
|  |     whiteList: {}, | ||||||
|  |     stripIgnoreTag: true, | ||||||
|  |     stripIgnoreTagBody: true, | ||||||
|  |   }); | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| import { consume } from "@lit/context"; |  | ||||||
| import { ResizeController } from "@lit-labs/observers/resize-controller"; | import { ResizeController } from "@lit-labs/observers/resize-controller"; | ||||||
|  | import { consume } from "@lit/context"; | ||||||
| import { mdiChevronDown, mdiChevronUp, mdiRestart } from "@mdi/js"; | import { mdiChevronDown, mdiChevronUp, mdiRestart } from "@mdi/js"; | ||||||
| import { differenceInMinutes } from "date-fns"; | import { differenceInMinutes } from "date-fns"; | ||||||
| import type { DataZoomComponentOption } from "echarts/components"; | import type { DataZoomComponentOption } from "echarts/components"; | ||||||
| @@ -7,15 +7,16 @@ import type { EChartsType } from "echarts/core"; | |||||||
| import type { | import type { | ||||||
|   ECElementEvent, |   ECElementEvent, | ||||||
|   LegendComponentOption, |   LegendComponentOption, | ||||||
|  |   LineSeriesOption, | ||||||
|   XAXisOption, |   XAXisOption, | ||||||
|   YAXisOption, |   YAXisOption, | ||||||
|   LineSeriesOption, |  | ||||||
| } from "echarts/types/dist/shared"; | } from "echarts/types/dist/shared"; | ||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } 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 { classMap } from "lit/directives/class-map"; | import { classMap } from "lit/directives/class-map"; | ||||||
| import { styleMap } from "lit/directives/style-map"; | import { styleMap } from "lit/directives/style-map"; | ||||||
|  | import { ensureArray } from "../../common/array/ensure-array"; | ||||||
| import { getAllGraphColors } from "../../common/color/colors"; | import { getAllGraphColors } from "../../common/color/colors"; | ||||||
| import { fireEvent } from "../../common/dom/fire_event"; | import { fireEvent } from "../../common/dom/fire_event"; | ||||||
| import { listenMediaQuery } from "../../common/dom/media_query"; | import { listenMediaQuery } from "../../common/dom/media_query"; | ||||||
| @@ -24,10 +25,10 @@ import type { Themes } from "../../data/ws-themes"; | |||||||
| import type { ECOption } from "../../resources/echarts"; | import type { ECOption } from "../../resources/echarts"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import { isMac } from "../../util/is_mac"; | import { isMac } from "../../util/is_mac"; | ||||||
| import "../ha-icon-button"; |  | ||||||
| import { formatTimeLabel } from "./axis-label"; |  | ||||||
| import { ensureArray } from "../../common/array/ensure-array"; |  | ||||||
| import "../chips/ha-assist-chip"; | import "../chips/ha-assist-chip"; | ||||||
|  | import "../ha-icon-button"; | ||||||
|  | import { filterXSS } from "../../common/util/xss"; | ||||||
|  | import { formatTimeLabel } from "./axis-label"; | ||||||
| import { downSampleLineData } from "./down-sample"; | import { downSampleLineData } from "./down-sample"; | ||||||
|  |  | ||||||
| export const MIN_TIME_BETWEEN_UPDATES = 60 * 5 * 1000; | export const MIN_TIME_BETWEEN_UPDATES = 60 * 5 * 1000; | ||||||
| @@ -63,6 +64,9 @@ export class HaChartBase extends LitElement { | |||||||
|   @property({ attribute: "small-controls", type: Boolean }) |   @property({ attribute: "small-controls", type: Boolean }) | ||||||
|   public smallControls?: boolean; |   public smallControls?: boolean; | ||||||
|  |  | ||||||
|  |   @property({ attribute: "hide-reset-button", type: Boolean }) | ||||||
|  |   public hideResetButton?: boolean; | ||||||
|  |  | ||||||
|   // extraComponents is not reactive and should not trigger updates |   // extraComponents is not reactive and should not trigger updates | ||||||
|   public extraComponents?: any[]; |   public extraComponents?: any[]; | ||||||
|  |  | ||||||
| @@ -215,7 +219,7 @@ export class HaChartBase extends LitElement { | |||||||
|         </div> |         </div> | ||||||
|         ${this._renderLegend()} |         ${this._renderLegend()} | ||||||
|         <div class="chart-controls ${classMap({ small: this.smallControls })}"> |         <div class="chart-controls ${classMap({ small: this.smallControls })}"> | ||||||
|           ${this._isZoomed |           ${this._isZoomed && !this.hideResetButton | ||||||
|             ? html`<ha-icon-button |             ? html`<ha-icon-button | ||||||
|                 class="zoom-reset" |                 class="zoom-reset" | ||||||
|                 .path=${mdiRestart} |                 .path=${mdiRestart} | ||||||
| @@ -353,20 +357,12 @@ export class HaChartBase extends LitElement { | |||||||
|  |  | ||||||
|       this.chart = echarts.init(container, "custom"); |       this.chart = echarts.init(container, "custom"); | ||||||
|       this.chart.on("datazoom", (e: any) => { |       this.chart.on("datazoom", (e: any) => { | ||||||
|         const { start, end } = e.batch?.[0] ?? e; |         this._handleDataZoomEvent(e); | ||||||
|         this._isZoomed = start !== 0 || end !== 100; |  | ||||||
|         this._zoomRatio = (end - start) / 100; |  | ||||||
|         if (this._isTouchDevice) { |  | ||||||
|           // zooming changes the axis pointer so we need to hide it |  | ||||||
|           this.chart?.dispatchAction({ |  | ||||||
|             type: "hideTip", |  | ||||||
|             from: "datazoom", |  | ||||||
|           }); |  | ||||||
|         } |  | ||||||
|       }); |       }); | ||||||
|       this.chart.on("click", (e: ECElementEvent) => { |       this.chart.on("click", (e: ECElementEvent) => { | ||||||
|         fireEvent(this, "chart-click", e); |         fireEvent(this, "chart-click", e); | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       if (!this.options?.dataZoom) { |       if (!this.options?.dataZoom) { | ||||||
|         this.chart.getZr().on("dblclick", this._handleClickZoom); |         this.chart.getZr().on("dblclick", this._handleClickZoom); | ||||||
|       } |       } | ||||||
| @@ -816,7 +812,8 @@ export class HaChartBase extends LitElement { | |||||||
|           }; |           }; | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|       return { ...s, data }; |       const name = filterXSS(String(s.name ?? s.id ?? "")); | ||||||
|  |       return { ...s, name, data }; | ||||||
|     }); |     }); | ||||||
|     return series as ECOption["series"]; |     return series as ECOption["series"]; | ||||||
|   } |   } | ||||||
| @@ -868,10 +865,60 @@ export class HaChartBase extends LitElement { | |||||||
|     }); |     }); | ||||||
|   }; |   }; | ||||||
|  |  | ||||||
|  |   public zoom(start: number, end: number, silent = false) { | ||||||
|  |     this.chart?.dispatchAction({ | ||||||
|  |       type: "dataZoom", | ||||||
|  |       start, | ||||||
|  |       end, | ||||||
|  |       silent, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private _handleZoomReset() { |   private _handleZoomReset() { | ||||||
|     this.chart?.dispatchAction({ type: "dataZoom", start: 0, end: 100 }); |     this.chart?.dispatchAction({ type: "dataZoom", start: 0, end: 100 }); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private _handleDataZoomEvent(e: any) { | ||||||
|  |     const zoomData = e.batch?.[0] ?? e; | ||||||
|  |     let start = typeof zoomData.start === "number" ? zoomData.start : 0; | ||||||
|  |     let end = typeof zoomData.end === "number" ? zoomData.end : 100; | ||||||
|  |  | ||||||
|  |     if ( | ||||||
|  |       start === 0 && | ||||||
|  |       end === 100 && | ||||||
|  |       zoomData.startValue !== undefined && | ||||||
|  |       zoomData.endValue !== undefined | ||||||
|  |     ) { | ||||||
|  |       const option = this.chart!.getOption(); | ||||||
|  |       const xAxis = option.xAxis?.[0] ?? option.xAxis; | ||||||
|  |  | ||||||
|  |       if (xAxis?.min && xAxis?.max) { | ||||||
|  |         const axisMin = new Date(xAxis.min).getTime(); | ||||||
|  |         const axisMax = new Date(xAxis.max).getTime(); | ||||||
|  |         const axisRange = axisMax - axisMin; | ||||||
|  |  | ||||||
|  |         start = Math.max( | ||||||
|  |           0, | ||||||
|  |           Math.min(100, ((zoomData.startValue - axisMin) / axisRange) * 100) | ||||||
|  |         ); | ||||||
|  |         end = Math.max( | ||||||
|  |           0, | ||||||
|  |           Math.min(100, ((zoomData.endValue - axisMin) / axisRange) * 100) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     this._isZoomed = start !== 0 || end !== 100; | ||||||
|  |     this._zoomRatio = (end - start) / 100; | ||||||
|  |     if (this._isTouchDevice) { | ||||||
|  |       this.chart?.dispatchAction({ | ||||||
|  |         type: "hideTip", | ||||||
|  |         from: "datazoom", | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |     fireEvent(this, "chart-zoom", { start, end }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private _legendClick(ev: any) { |   private _legendClick(ev: any) { | ||||||
|     if (!this.chart) { |     if (!this.chart) { | ||||||
|       return; |       return; | ||||||
| @@ -929,7 +976,7 @@ export class HaChartBase extends LitElement { | |||||||
|       right: 4px; |       right: 4px; | ||||||
|       display: flex; |       display: flex; | ||||||
|       flex-direction: column; |       flex-direction: column; | ||||||
|       gap: 4px; |       gap: var(--ha-space-1); | ||||||
|     } |     } | ||||||
|     .chart-controls.small { |     .chart-controls.small { | ||||||
|       top: 0; |       top: 0; | ||||||
| @@ -938,7 +985,7 @@ export class HaChartBase extends LitElement { | |||||||
|     .chart-controls ha-icon-button, |     .chart-controls ha-icon-button, | ||||||
|     .chart-controls ::slotted(ha-icon-button) { |     .chart-controls ::slotted(ha-icon-button) { | ||||||
|       background: var(--card-background-color); |       background: var(--card-background-color); | ||||||
|       border-radius: 4px; |       border-radius: var(--ha-border-radius-sm); | ||||||
|       --mdc-icon-button-size: 32px; |       --mdc-icon-button-size: 32px; | ||||||
|       color: var(--primary-color); |       color: var(--primary-color); | ||||||
|       border: 1px solid var(--divider-color); |       border: 1px solid var(--divider-color); | ||||||
| @@ -966,7 +1013,7 @@ export class HaChartBase extends LitElement { | |||||||
|       flex-wrap: wrap; |       flex-wrap: wrap; | ||||||
|       justify-content: center; |       justify-content: center; | ||||||
|       align-items: center; |       align-items: center; | ||||||
|       gap: 8px; |       gap: var(--ha-space-2); | ||||||
|     } |     } | ||||||
|     .chart-legend li { |     .chart-legend li { | ||||||
|       height: 24px; |       height: 24px; | ||||||
| @@ -991,7 +1038,7 @@ export class HaChartBase extends LitElement { | |||||||
|     .chart-legend .bullet { |     .chart-legend .bullet { | ||||||
|       border-width: 1px; |       border-width: 1px; | ||||||
|       border-style: solid; |       border-style: solid; | ||||||
|       border-radius: 50%; |       border-radius: var(--ha-border-radius-circle); | ||||||
|       display: block; |       display: block; | ||||||
|       height: 16px; |       height: 16px; | ||||||
|       width: 16px; |       width: 16px; | ||||||
| @@ -1024,5 +1071,9 @@ declare global { | |||||||
|     "dataset-hidden": { id: string }; |     "dataset-hidden": { id: string }; | ||||||
|     "dataset-unhidden": { id: string }; |     "dataset-unhidden": { id: string }; | ||||||
|     "chart-click": ECElementEvent; |     "chart-click": ECElementEvent; | ||||||
|  |     "chart-zoom": { | ||||||
|  |       start: number; | ||||||
|  |       end: number; | ||||||
|  |     }; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ import { ResizeController } from "@lit-labs/observers/resize-controller"; | |||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
| import type { ECOption } from "../../resources/echarts"; | import type { ECOption } from "../../resources/echarts"; | ||||||
| import { measureTextWidth } from "../../util/text"; | import { measureTextWidth } from "../../util/text"; | ||||||
|  | import { filterXSS } from "../../common/util/xss"; | ||||||
| import "./ha-chart-base"; | import "./ha-chart-base"; | ||||||
| import { NODE_SIZE } from "../trace/hat-graph-const"; | import { NODE_SIZE } from "../trace/hat-graph-const"; | ||||||
| import "../ha-alert"; | import "../ha-alert"; | ||||||
| @@ -92,12 +93,12 @@ export class HaSankeyChart extends LitElement { | |||||||
|       : data.value; |       : data.value; | ||||||
|     if (data.id) { |     if (data.id) { | ||||||
|       const node = this.data.nodes.find((n) => n.id === data.id); |       const node = this.data.nodes.find((n) => n.id === data.id); | ||||||
|       return `${params.marker} ${node?.label ?? data.id}<br>${value}`; |       return `${params.marker} ${filterXSS(node?.label ?? data.id)}<br>${value}`; | ||||||
|     } |     } | ||||||
|     if (data.source && data.target) { |     if (data.source && data.target) { | ||||||
|       const source = this.data.nodes.find((n) => n.id === data.source); |       const source = this.data.nodes.find((n) => n.id === data.source); | ||||||
|       const target = this.data.nodes.find((n) => n.id === data.target); |       const target = this.data.nodes.find((n) => n.id === data.target); | ||||||
|       return `${source?.label ?? data.source} → ${target?.label ?? data.target}<br>${value}`; |       return `${filterXSS(source?.label ?? data.source)} → ${filterXSS(target?.label ?? data.target)}<br>${value}`; | ||||||
|     } |     } | ||||||
|     return null; |     return null; | ||||||
|   }; |   }; | ||||||
|   | |||||||
| @@ -66,6 +66,9 @@ export class StateHistoryChartLine extends LitElement { | |||||||
|   @property({ attribute: "expand-legend", type: Boolean }) |   @property({ attribute: "expand-legend", type: Boolean }) | ||||||
|   public expandLegend?: boolean; |   public expandLegend?: boolean; | ||||||
|  |  | ||||||
|  |   @property({ attribute: "hide-reset-button", type: Boolean }) | ||||||
|  |   public hideResetButton?: boolean; | ||||||
|  |  | ||||||
|   @state() private _chartData: LineSeriesOption[] = []; |   @state() private _chartData: LineSeriesOption[] = []; | ||||||
|  |  | ||||||
|   @state() private _entityIds: string[] = []; |   @state() private _entityIds: string[] = []; | ||||||
| @@ -94,7 +97,9 @@ export class StateHistoryChartLine extends LitElement { | |||||||
|         style=${styleMap({ height: this.height })} |         style=${styleMap({ height: this.height })} | ||||||
|         @dataset-hidden=${this._datasetHidden} |         @dataset-hidden=${this._datasetHidden} | ||||||
|         @dataset-unhidden=${this._datasetUnhidden} |         @dataset-unhidden=${this._datasetUnhidden} | ||||||
|  |         @chart-zoom=${this._handleDataZoom} | ||||||
|         .expandLegend=${this.expandLegend} |         .expandLegend=${this.expandLegend} | ||||||
|  |         .hideResetButton=${this.hideResetButton} | ||||||
|       ></ha-chart-base> |       ></ha-chart-base> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| @@ -192,6 +197,19 @@ export class StateHistoryChartLine extends LitElement { | |||||||
|     this._hiddenStats.delete(ev.detail.id); |     this._hiddenStats.delete(ev.detail.id); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public zoom(start: number, end: number) { | ||||||
|  |     const chartBase = this.shadowRoot!.querySelector("ha-chart-base")!; | ||||||
|  |     chartBase.zoom(start, end, true); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _handleDataZoom(ev: CustomEvent) { | ||||||
|  |     fireEvent(this, "chart-zoom-with-index", { | ||||||
|  |       start: ev.detail.start ?? 0, | ||||||
|  |       end: ev.detail.end ?? 100, | ||||||
|  |       chartIndex: this.chartIndex, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   public willUpdate(changedProps: PropertyValues) { |   public willUpdate(changedProps: PropertyValues) { | ||||||
|     if ( |     if ( | ||||||
|       changedProps.has("data") || |       changedProps.has("data") || | ||||||
|   | |||||||
| @@ -51,6 +51,9 @@ export class StateHistoryChartTimeline extends LitElement { | |||||||
|  |  | ||||||
|   @property({ attribute: false, type: Number }) public chartIndex?; |   @property({ attribute: false, type: Number }) public chartIndex?; | ||||||
|  |  | ||||||
|  |   @property({ attribute: "hide-reset-button", type: Boolean }) | ||||||
|  |   public hideResetButton?: boolean; | ||||||
|  |  | ||||||
|   @state() private _chartData: CustomSeriesOption[] = []; |   @state() private _chartData: CustomSeriesOption[] = []; | ||||||
|  |  | ||||||
|   @state() private _chartOptions?: ECOption; |   @state() private _chartOptions?: ECOption; | ||||||
| @@ -68,6 +71,8 @@ export class StateHistoryChartTimeline extends LitElement { | |||||||
|         .data=${this._chartData as ECOption["series"]} |         .data=${this._chartData as ECOption["series"]} | ||||||
|         small-controls |         small-controls | ||||||
|         @chart-click=${this._handleChartClick} |         @chart-click=${this._handleChartClick} | ||||||
|  |         @chart-zoom=${this._handleDataZoom} | ||||||
|  |         .hideResetButton=${this.hideResetButton} | ||||||
|       ></ha-chart-base> |       ></ha-chart-base> | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
| @@ -256,6 +261,19 @@ export class StateHistoryChartTimeline extends LitElement { | |||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   public zoom(start: number, end: number) { | ||||||
|  |     const chartBase = this.shadowRoot!.querySelector("ha-chart-base")!; | ||||||
|  |     chartBase.zoom(start, end, true); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _handleDataZoom(ev: CustomEvent) { | ||||||
|  |     fireEvent(this, "chart-zoom-with-index", { | ||||||
|  |       start: ev.detail.start ?? 0, | ||||||
|  |       end: ev.detail.end ?? 100, | ||||||
|  |       chartIndex: this.chartIndex, | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private _generateData() { |   private _generateData() { | ||||||
|     const computedStyles = getComputedStyle(this); |     const computedStyles = getComputedStyle(this); | ||||||
|     let stateHistory = this.data; |     let stateHistory = this.data; | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| import type { PropertyValues } from "lit"; | import type { PropertyValues } from "lit"; | ||||||
| import { css, html, LitElement } from "lit"; | import { css, html, LitElement, nothing } from "lit"; | ||||||
| import { customElement, eventOptions, property, state } from "lit/decorators"; | import { customElement, eventOptions, property, state } from "lit/decorators"; | ||||||
| import type { RenderItemFunction } from "@lit-labs/virtualizer/virtualize"; | import type { RenderItemFunction } from "@lit-labs/virtualizer/virtualize"; | ||||||
|  | import { mdiRestart } from "@mdi/js"; | ||||||
| import { isComponentLoaded } from "../../common/config/is_component_loaded"; | import { isComponentLoaded } from "../../common/config/is_component_loaded"; | ||||||
| import { restoreScroll } from "../../common/decorators/restore-scroll"; | import { restoreScroll } from "../../common/decorators/restore-scroll"; | ||||||
| import type { | import type { | ||||||
| @@ -11,6 +12,10 @@ import type { | |||||||
| } from "../../data/history"; | } from "../../data/history"; | ||||||
| import { loadVirtualizer } from "../../resources/virtualizer"; | import { loadVirtualizer } from "../../resources/virtualizer"; | ||||||
| import type { HomeAssistant } from "../../types"; | import type { HomeAssistant } from "../../types"; | ||||||
|  | import type { StateHistoryChartLine } from "./state-history-chart-line"; | ||||||
|  | import type { StateHistoryChartTimeline } from "./state-history-chart-timeline"; | ||||||
|  | import "../ha-fab"; | ||||||
|  | import "../ha-svg-icon"; | ||||||
| import "./state-history-chart-line"; | import "./state-history-chart-line"; | ||||||
| import "./state-history-chart-timeline"; | import "./state-history-chart-timeline"; | ||||||
|  |  | ||||||
| @@ -29,6 +34,11 @@ const chunkData = (inputArray: any[], chunks: number) => | |||||||
| declare global { | declare global { | ||||||
|   interface HASSDomEvents { |   interface HASSDomEvents { | ||||||
|     "y-width-changed": { value: number; chartIndex: number }; |     "y-width-changed": { value: number; chartIndex: number }; | ||||||
|  |     "chart-zoom-with-index": { | ||||||
|  |       start: number; | ||||||
|  |       end: number; | ||||||
|  |       chartIndex: number; | ||||||
|  |     }; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -74,6 +84,9 @@ export class StateHistoryCharts extends LitElement { | |||||||
|   @property({ attribute: "expand-legend", type: Boolean }) |   @property({ attribute: "expand-legend", type: Boolean }) | ||||||
|   public expandLegend?: boolean; |   public expandLegend?: boolean; | ||||||
|  |  | ||||||
|  |   @property({ attribute: "sync-charts", type: Boolean }) | ||||||
|  |   public syncCharts = false; | ||||||
|  |  | ||||||
|   private _computedStartTime!: Date; |   private _computedStartTime!: Date; | ||||||
|  |  | ||||||
|   private _computedEndTime!: Date; |   private _computedEndTime!: Date; | ||||||
| @@ -84,6 +97,10 @@ export class StateHistoryCharts extends LitElement { | |||||||
|  |  | ||||||
|   @state() private _chartCount = 0; |   @state() private _chartCount = 0; | ||||||
|  |  | ||||||
|  |   @state() private _hasZoomedCharts = false; | ||||||
|  |  | ||||||
|  |   private _isSyncing = false; | ||||||
|  |  | ||||||
|   // @ts-ignore |   // @ts-ignore | ||||||
|   @restoreScroll(".container") private _savedScrollPos?: number; |   @restoreScroll(".container") private _savedScrollPos?: number; | ||||||
|  |  | ||||||
| @@ -115,8 +132,12 @@ export class StateHistoryCharts extends LitElement { | |||||||
|     // eslint-disable-next-line lit/no-this-assign-in-render |     // eslint-disable-next-line lit/no-this-assign-in-render | ||||||
|     this._chartCount = combinedItems.length; |     this._chartCount = combinedItems.length; | ||||||
|  |  | ||||||
|     return this.virtualize |     return html` | ||||||
|       ? html`<div class="container ha-scrollbar" @scroll=${this._saveScrollPos}> |       ${this.virtualize | ||||||
|  |         ? html`<div | ||||||
|  |             class="container ha-scrollbar" | ||||||
|  |             @scroll=${this._saveScrollPos} | ||||||
|  |           > | ||||||
|             <lit-virtualizer |             <lit-virtualizer | ||||||
|               scroller |               scroller | ||||||
|               class="ha-scrollbar" |               class="ha-scrollbar" | ||||||
| @@ -127,7 +148,20 @@ export class StateHistoryCharts extends LitElement { | |||||||
|           </div>` |           </div>` | ||||||
|         : html`${combinedItems.map((item, index) => |         : html`${combinedItems.map((item, index) => | ||||||
|             this._renderHistoryItem(item, index) |             this._renderHistoryItem(item, index) | ||||||
|         )}`; |           )}`} | ||||||
|  |       ${this.syncCharts && this._hasZoomedCharts | ||||||
|  |         ? html`<ha-fab | ||||||
|  |             slot="fab" | ||||||
|  |             class="reset-button" | ||||||
|  |             .label=${this.hass.localize( | ||||||
|  |               "ui.components.history_charts.zoom_reset" | ||||||
|  |             )} | ||||||
|  |             @click=${this._handleGlobalZoomReset} | ||||||
|  |           > | ||||||
|  |             <ha-svg-icon slot="icon" .path=${mdiRestart}></ha-svg-icon> | ||||||
|  |           </ha-fab>` | ||||||
|  |         : nothing} | ||||||
|  |     `; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   private _renderHistoryItem: RenderItemFunction< |   private _renderHistoryItem: RenderItemFunction< | ||||||
| @@ -156,8 +190,10 @@ export class StateHistoryCharts extends LitElement { | |||||||
|           .maxYAxis=${this.maxYAxis} |           .maxYAxis=${this.maxYAxis} | ||||||
|           .fitYData=${this.fitYData} |           .fitYData=${this.fitYData} | ||||||
|           @y-width-changed=${this._yWidthChanged} |           @y-width-changed=${this._yWidthChanged} | ||||||
|  |           @chart-zoom-with-index=${this._handleTimelineSync} | ||||||
|           .height=${this.virtualize ? undefined : this.height} |           .height=${this.virtualize ? undefined : this.height} | ||||||
|           .expandLegend=${this.expandLegend} |           .expandLegend=${this.expandLegend} | ||||||
|  |           ?hide-reset-button=${this.syncCharts} | ||||||
|         ></state-history-chart-line> |         ></state-history-chart-line> | ||||||
|       </div> `; |       </div> `; | ||||||
|     } |     } | ||||||
| @@ -175,6 +211,8 @@ export class StateHistoryCharts extends LitElement { | |||||||
|         .chartIndex=${index} |         .chartIndex=${index} | ||||||
|         .clickForMoreInfo=${this.clickForMoreInfo} |         .clickForMoreInfo=${this.clickForMoreInfo} | ||||||
|         @y-width-changed=${this._yWidthChanged} |         @y-width-changed=${this._yWidthChanged} | ||||||
|  |         @chart-zoom-with-index=${this._handleTimelineSync} | ||||||
|  |         ?hide-reset-button=${this.syncCharts} | ||||||
|       ></state-history-chart-timeline> |       ></state-history-chart-timeline> | ||||||
|     </div> `; |     </div> `; | ||||||
|   }; |   }; | ||||||
| @@ -264,6 +302,66 @@ export class StateHistoryCharts extends LitElement { | |||||||
|     this._maxYWidth = Math.max(...Object.values(this._childYWidths), 0); |     this._maxYWidth = Math.max(...Object.values(this._childYWidths), 0); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   private _handleTimelineSync( | ||||||
|  |     e: CustomEvent<HASSDomEvents["chart-zoom-with-index"]> | ||||||
|  |   ) { | ||||||
|  |     if (!this.syncCharts || this._isSyncing) { | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const { start, end, chartIndex } = e.detail; | ||||||
|  |  | ||||||
|  |     this._hasZoomedCharts = start !== 0 || end !== 100; | ||||||
|  |     this._syncZoomToAllCharts(start, end, chartIndex); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _syncZoomToAllCharts( | ||||||
|  |     start: number, | ||||||
|  |     end: number, | ||||||
|  |     sourceChartIndex?: number | ||||||
|  |   ) { | ||||||
|  |     this._isSyncing = true; | ||||||
|  |  | ||||||
|  |     requestAnimationFrame(() => { | ||||||
|  |       const chartComponents = this.renderRoot.querySelectorAll( | ||||||
|  |         "state-history-chart-line, state-history-chart-timeline" | ||||||
|  |       ) as unknown as (StateHistoryChartLine | StateHistoryChartTimeline)[]; | ||||||
|  |  | ||||||
|  |       chartComponents.forEach((chartComponent, index) => { | ||||||
|  |         if (index === sourceChartIndex) { | ||||||
|  |           return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if ("zoom" in chartComponent) { | ||||||
|  |           chartComponent.zoom(start, end); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       this._isSyncing = false; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   private _handleGlobalZoomReset() { | ||||||
|  |     this._hasZoomedCharts = false; | ||||||
|  |     this._isSyncing = true; | ||||||
|  |  | ||||||
|  |     requestAnimationFrame(() => { | ||||||
|  |       const chartComponents = this.renderRoot.querySelectorAll( | ||||||
|  |         "state-history-chart-line, state-history-chart-timeline" | ||||||
|  |       ); | ||||||
|  |  | ||||||
|  |       chartComponents.forEach((chartComponent: any) => { | ||||||
|  |         const chartBase = | ||||||
|  |           chartComponent.renderRoot?.querySelector("ha-chart-base"); | ||||||
|  |  | ||||||
|  |         if (chartBase && chartBase.chart) { | ||||||
|  |           chartBase.zoom(0, 100); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |       this._isSyncing = false; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|   private _isHistoryEmpty(): boolean { |   private _isHistoryEmpty(): boolean { | ||||||
|     const historyDataEmpty = |     const historyDataEmpty = | ||||||
|       !this.historyData || |       !this.historyData || | ||||||
| @@ -345,6 +443,12 @@ export class StateHistoryCharts extends LitElement { | |||||||
|     state-history-chart-line { |     state-history-chart-line { | ||||||
|       width: 100%; |       width: 100%; | ||||||
|     } |     } | ||||||
|  |     .reset-button { | ||||||
|  |       position: fixed; | ||||||
|  |       bottom: calc(24px + var(--safe-area-inset-bottom)); | ||||||
|  |       right: calc(24px + var(--safe-area-inset-bottom)); | ||||||
|  |       z-index: 1; | ||||||
|  |     } | ||||||
|   `; |   `; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,6 +6,8 @@ import { computeDomain } from "../../common/entity/compute_domain"; | |||||||
| import { stateColorProperties } from "../../common/entity/state_color"; | import { stateColorProperties } from "../../common/entity/state_color"; | ||||||
| import { UNAVAILABLE, UNKNOWN } from "../../data/entity"; | import { UNAVAILABLE, UNKNOWN } from "../../data/entity"; | ||||||
| import { computeCssValue } from "../../resources/css-variables"; | import { computeCssValue } from "../../resources/css-variables"; | ||||||
|  | import { computeStateDomain } from "../../common/entity/compute_state_domain"; | ||||||
|  | import { FIXED_DOMAIN_STATES } from "../../common/entity/get_states"; | ||||||
|  |  | ||||||
| const DOMAIN_STATE_SHADES: Record<string, Record<string, number>> = { | const DOMAIN_STATE_SHADES: Record<string, Record<string, number>> = { | ||||||
|   media_player: { |   media_player: { | ||||||
| @@ -51,6 +53,28 @@ function computeTimelineStateColor( | |||||||
| let colorIndex = 0; | let colorIndex = 0; | ||||||
| const stateColorMap = new Map<string, string>(); | const stateColorMap = new Map<string, string>(); | ||||||
|  |  | ||||||
|  | function computeTimelineEnumColor( | ||||||
|  |   state: string, | ||||||
|  |   computedStyles: CSSStyleDeclaration, | ||||||
|  |   stateObj?: HassEntity | ||||||
|  | ): string | undefined { | ||||||
|  |   if (!stateObj) { | ||||||
|  |     return undefined; | ||||||
|  |   } | ||||||
|  |   const domain = computeStateDomain(stateObj); | ||||||
|  |   const states = | ||||||
|  |     FIXED_DOMAIN_STATES[domain] || | ||||||
|  |     (domain === "sensor" && | ||||||
|  |       stateObj.attributes.device_class === "enum" && | ||||||
|  |       stateObj.attributes.options) || | ||||||
|  |     []; | ||||||
|  |   const idx = states.indexOf(state); | ||||||
|  |   if (idx === -1) { | ||||||
|  |     return undefined; | ||||||
|  |   } | ||||||
|  |   return getGraphColorByIndex(idx, computedStyles); | ||||||
|  | } | ||||||
|  |  | ||||||
| function computeTimeLineGenericColor( | function computeTimeLineGenericColor( | ||||||
|   state: string, |   state: string, | ||||||
|   computedStyles: CSSStyleDeclaration |   computedStyles: CSSStyleDeclaration | ||||||
| @@ -71,6 +95,7 @@ export function computeTimelineColor( | |||||||
| ): string { | ): string { | ||||||
|   return ( |   return ( | ||||||
|     computeTimelineStateColor(state, computedStyles, stateObj) || |     computeTimelineStateColor(state, computedStyles, stateObj) || | ||||||
|  |     computeTimelineEnumColor(state, computedStyles, stateObj) || | ||||||
|     computeTimeLineGenericColor(state, computedStyles) |     computeTimeLineGenericColor(state, computedStyles) | ||||||
|   ); |   ); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -290,7 +290,9 @@ export class DialogDataTableSettings extends LitElement { | |||||||
|           ha-dialog { |           ha-dialog { | ||||||
|             --vertical-align-dialog: flex-start; |             --vertical-align-dialog: flex-start; | ||||||
|             --dialog-surface-margin-top: 250px; |             --dialog-surface-margin-top: 250px; | ||||||
|             --ha-dialog-border-radius: 28px 28px 0 0; |             --ha-dialog-border-radius: var(--ha-border-radius-4xl) | ||||||
|  |               var(--ha-border-radius-4xl) var(--ha-border-radius-square) | ||||||
|  |               var(--ha-border-radius-square); | ||||||
|             --mdc-dialog-min-height: calc(100% - 250px); |             --mdc-dialog-min-height: calc(100% - 250px); | ||||||
|             --mdc-dialog-max-height: calc(100% - 250px); |             --mdc-dialog-max-height: calc(100% - 250px); | ||||||
|           } |           } | ||||||
|   | |||||||
| @@ -12,9 +12,8 @@ class HaDataTableIcon extends LitElement { | |||||||
|  |  | ||||||
|   protected render(): TemplateResult { |   protected render(): TemplateResult { | ||||||
|     return html` |     return html` | ||||||
|       <ha-tooltip .content=${this.tooltip}> |       <ha-tooltip for="svg-icon">${this.tooltip}</ha-tooltip> | ||||||
|         <ha-svg-icon .path=${this.path}></ha-svg-icon> |       <ha-svg-icon id="svg-icon" .path=${this.path}></ha-svg-icon> | ||||||
|       </ha-tooltip> |  | ||||||
|     `; |     `; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1053,7 +1053,7 @@ export class HaDataTable extends LitElement { | |||||||
|  |  | ||||||
|         .mdc-data-table { |         .mdc-data-table { | ||||||
|           background-color: var(--data-table-background-color); |           background-color: var(--data-table-background-color); | ||||||
|           border-radius: 4px; |           border-radius: var(--ha-border-radius-sm); | ||||||
|           border-width: 1px; |           border-width: 1px; | ||||||
|           border-style: solid; |           border-style: solid; | ||||||
|           border-color: var(--divider-color); |           border-color: var(--divider-color); | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user