mirror of
https://github.com/home-assistant/frontend.git
synced 2025-08-12 18:59:26 +00:00
Compare commits
506 Commits
remove-unc
...
fix/Dropdo
Author | SHA1 | Date | |
---|---|---|---|
![]() |
35d6c638ab | ||
![]() |
68f8239708 | ||
![]() |
0db64cca0b | ||
![]() |
accfda5f4b | ||
![]() |
c97c20f57d | ||
![]() |
2725d0191d | ||
![]() |
852cc62398 | ||
![]() |
654e3ce437 | ||
![]() |
20a3a00aec | ||
![]() |
22b927d666 | ||
![]() |
709d6be2e3 | ||
![]() |
fbda9ca418 | ||
![]() |
4e97e3763e | ||
![]() |
4c9c52d27d | ||
![]() |
87bcd3e471 | ||
![]() |
7e9b01b56d | ||
![]() |
713763fc21 | ||
![]() |
5b7ab1bfcb | ||
![]() |
4b0d19b615 | ||
![]() |
90e5d259af | ||
![]() |
af3a331f57 | ||
![]() |
67c60a4aa8 | ||
![]() |
62de16bb8e | ||
![]() |
d9b71e754d | ||
![]() |
5fc950f09f | ||
![]() |
0725c7b160 | ||
![]() |
469dbbcccc | ||
![]() |
ffdd661b1f | ||
![]() |
81922f5a3e | ||
![]() |
7e25366897 | ||
![]() |
8ab61b5468 | ||
![]() |
8239f6dd60 | ||
![]() |
45dce18e4d | ||
![]() |
a428ad0655 | ||
![]() |
1b54d51e4a | ||
![]() |
eb1354d229 | ||
![]() |
4d21f9e80c | ||
![]() |
62f46baacf | ||
![]() |
a3090796d2 | ||
![]() |
c34c5d64f9 | ||
![]() |
66228f5858 | ||
![]() |
ac378cfe6d | ||
![]() |
7ecf8b755e | ||
![]() |
141107f1f3 | ||
![]() |
b5277dee53 | ||
![]() |
4b593c1c96 | ||
![]() |
50ce1b94c8 | ||
![]() |
8bf27a83ec | ||
![]() |
389f0d3d23 | ||
![]() |
b966601e6a | ||
![]() |
f2a0881821 | ||
![]() |
50a49eae43 | ||
![]() |
1c04561004 | ||
![]() |
b80d94d260 | ||
![]() |
87012e23e7 | ||
![]() |
f39758b103 | ||
![]() |
697bbf428e | ||
![]() |
c7444a2605 | ||
![]() |
3a5f4d33d2 | ||
![]() |
c3dc62523b | ||
![]() |
424622061a | ||
![]() |
a3b021b11d | ||
![]() |
b60ad8b143 | ||
![]() |
e376efc579 | ||
![]() |
382035a1d4 | ||
![]() |
542e22fe0e | ||
![]() |
af37d57779 | ||
![]() |
fbef0b0186 | ||
![]() |
9e67d6add8 | ||
![]() |
25c702ad2b | ||
![]() |
6516597c93 | ||
![]() |
1df9c38a8c | ||
![]() |
bd7217145a | ||
![]() |
569fef38a4 | ||
![]() |
f21c89cf1a | ||
![]() |
02cc418969 | ||
![]() |
4faba159c0 | ||
![]() |
29816e6c5e | ||
![]() |
5317a11c39 | ||
![]() |
27c53b3241 | ||
![]() |
919befa961 | ||
![]() |
f9c02ed099 | ||
![]() |
b35c325f43 | ||
![]() |
b82f1128fe | ||
![]() |
178feb7330 | ||
![]() |
0118a5bf4c | ||
![]() |
e0087bd142 | ||
![]() |
c2d3e7900e | ||
![]() |
fb8312110b | ||
![]() |
16de57342e | ||
![]() |
ad6e041c04 | ||
![]() |
e22e3e88a0 | ||
![]() |
dc8a50965c | ||
![]() |
1914de7ddf | ||
![]() |
2e505cfb1f | ||
![]() |
ab49aca815 | ||
![]() |
c96968e476 | ||
![]() |
8f050516ec | ||
![]() |
27d2b244a4 | ||
![]() |
be2f2c6271 | ||
![]() |
8dc2797b16 | ||
![]() |
7ca8dabc44 | ||
![]() |
baeb55e217 | ||
![]() |
a8502fcc11 | ||
![]() |
9f5bc5b196 | ||
![]() |
7556ab9506 | ||
![]() |
bf176ac314 | ||
![]() |
9903e22eaa | ||
![]() |
1e0f7d9629 | ||
![]() |
e8a140af44 | ||
![]() |
b091d4f298 | ||
![]() |
35cf3063cb | ||
![]() |
7141ef17be | ||
![]() |
be2c68c0bb | ||
![]() |
7c944d3767 | ||
![]() |
1d4f02df2e | ||
![]() |
2007a74a20 | ||
![]() |
8c0839ad57 | ||
![]() |
516b9a54c4 | ||
![]() |
0d3e730c9c | ||
![]() |
c7a87d02b2 | ||
![]() |
dd082c204b | ||
![]() |
c4af3d1579 | ||
![]() |
10eadbcbbb | ||
![]() |
17141824f7 | ||
![]() |
4cfd6c010f | ||
![]() |
daa9024bff | ||
![]() |
e96aca90fe | ||
![]() |
0580a31961 | ||
![]() |
5c42c5130c | ||
![]() |
72d1e37a23 | ||
![]() |
61c9072a08 | ||
![]() |
08b25f9c2a | ||
![]() |
1a03b49700 | ||
![]() |
2d4a8e2e45 | ||
![]() |
8486377604 | ||
![]() |
3a4e9b6856 | ||
![]() |
5f5ac5419b | ||
![]() |
92b7a3b477 | ||
![]() |
4326519a3f | ||
![]() |
00837acdfc | ||
![]() |
7704be12b1 | ||
![]() |
712ddb531b | ||
![]() |
d52afc3f71 | ||
![]() |
92f6083e0b | ||
![]() |
5751fdbe56 | ||
![]() |
962b30adb9 | ||
![]() |
3b5b3f3bb6 | ||
![]() |
1a6d96cf3a | ||
![]() |
034fd9b4df | ||
![]() |
eb79a1e7d7 | ||
![]() |
e25d4f17aa | ||
![]() |
ccde9cceee | ||
![]() |
578d3c4260 | ||
![]() |
bfdc9a3d86 | ||
![]() |
5315545a4d | ||
![]() |
82a3b9d80f | ||
![]() |
3de985a3b8 | ||
![]() |
567ee8000d | ||
![]() |
03939001b2 | ||
![]() |
30d18050d1 | ||
![]() |
95caf8c7df | ||
![]() |
6c1f328d71 | ||
![]() |
bb20ab8c2c | ||
![]() |
29eb73176a | ||
![]() |
17ad3a87f3 | ||
![]() |
ed7c9c33b9 | ||
![]() |
59b66219cb | ||
![]() |
1e2c1d1464 | ||
![]() |
5b86b1277f | ||
![]() |
41fdf31e34 | ||
![]() |
9bef5c2af9 | ||
![]() |
ed1a69071b | ||
![]() |
56d328b4db | ||
![]() |
33c7e0fa2d | ||
![]() |
4f1cf1110f | ||
![]() |
a434bfd944 | ||
![]() |
21ed8e4206 | ||
![]() |
169d782580 | ||
![]() |
8a015f4e38 | ||
![]() |
cbb08c6202 | ||
![]() |
6301bc713c | ||
![]() |
a5d7043ce4 | ||
![]() |
912d2cbd79 | ||
![]() |
48ee3a34eb | ||
![]() |
21263a1ffb | ||
![]() |
db59e138e9 | ||
![]() |
bc8012dcc9 | ||
![]() |
d8b43597a0 | ||
![]() |
d3bf0da289 | ||
![]() |
871949e760 | ||
![]() |
4fb42d3545 | ||
![]() |
2e58d6656c | ||
![]() |
a3024b38e9 | ||
![]() |
85f2016371 | ||
![]() |
1ce3347c2e | ||
![]() |
4f8415e8a7 | ||
![]() |
b202a36feb | ||
![]() |
7e3e224746 | ||
![]() |
503a7979d0 | ||
![]() |
f3ba6e7996 | ||
![]() |
f13dcb4139 | ||
![]() |
e8dc61ec36 | ||
![]() |
88c59c5c13 | ||
![]() |
fd06d434f2 | ||
![]() |
85f80ff863 | ||
![]() |
d56abe6b72 | ||
![]() |
d24d29e42f | ||
![]() |
bc14b8468d | ||
![]() |
f924f81ec1 | ||
![]() |
3a6382df55 | ||
![]() |
1dba049038 | ||
![]() |
f539516252 | ||
![]() |
abd02eda0f | ||
![]() |
99695d6cb3 | ||
![]() |
cb1c2b59df | ||
![]() |
8368f977b9 | ||
![]() |
e05595f318 | ||
![]() |
11cf2ec39d | ||
![]() |
e5c43fcfcd | ||
![]() |
520581c165 | ||
![]() |
d1119a3b61 | ||
![]() |
5dd029cc05 | ||
![]() |
510e010f97 | ||
![]() |
1300cffa3b | ||
![]() |
8fbcbb0b68 | ||
![]() |
e02a47a16a | ||
![]() |
7b26c1ffcb | ||
![]() |
d3e62454a5 | ||
![]() |
6b8f4e92a7 | ||
![]() |
b590b21183 | ||
![]() |
a08484f450 | ||
![]() |
2978ca13c5 | ||
![]() |
31c0850b14 | ||
![]() |
1d85f0717a | ||
![]() |
795c16a941 | ||
![]() |
55c8589841 | ||
![]() |
4687add37a | ||
![]() |
c25e23ccd6 | ||
![]() |
e42ddb8f0f | ||
![]() |
705c0e58fc | ||
![]() |
7427e17926 | ||
![]() |
2c4b31dcaa | ||
![]() |
ae8671af96 | ||
![]() |
f5ff55abc5 | ||
![]() |
b662512995 | ||
![]() |
64c3fb1723 | ||
![]() |
fb99dc4cd0 | ||
![]() |
e08a0c44ba | ||
![]() |
68935d46ce | ||
![]() |
141c8c5192 | ||
![]() |
7ca5467f4c | ||
![]() |
5de53964d9 | ||
![]() |
8d8807e659 | ||
![]() |
9347944cbd | ||
![]() |
480448acbb | ||
![]() |
202fa82646 | ||
![]() |
feecc9f838 | ||
![]() |
2f9e667517 | ||
![]() |
5547bc7356 | ||
![]() |
eb4ae926b7 | ||
![]() |
b239ec2b71 | ||
![]() |
e9cac94aee | ||
![]() |
5289cd3af1 | ||
![]() |
45a5c1c235 | ||
![]() |
db3709952c | ||
![]() |
447932eedb | ||
![]() |
10cc3bdd3f | ||
![]() |
6ee2bfed36 | ||
![]() |
01efb831b7 | ||
![]() |
9e1e20bd94 | ||
![]() |
869ace74ad | ||
![]() |
94d56367fc | ||
![]() |
68a5ba668e | ||
![]() |
b2b590cf67 | ||
![]() |
6f7c071769 | ||
![]() |
c1a7164ce7 | ||
![]() |
b77839c139 | ||
![]() |
e2f2a9322c | ||
![]() |
e4bd6c885d | ||
![]() |
8201701d17 | ||
![]() |
a5e6b78e1d | ||
![]() |
027eccba06 | ||
![]() |
12f10513f0 | ||
![]() |
9907ed51f0 | ||
![]() |
90e9f79841 | ||
![]() |
c30b9cdfcf | ||
![]() |
7e1fa0cf38 | ||
![]() |
b6587488d4 | ||
![]() |
552eeeddf6 | ||
![]() |
cbc150bad2 | ||
![]() |
8a4ed121b5 | ||
![]() |
a9793dc0a5 | ||
![]() |
c9deef84ca | ||
![]() |
1582aaeb4c | ||
![]() |
707520c15c | ||
![]() |
d5de435f06 | ||
![]() |
9e3dfaa400 | ||
![]() |
7aa92ec249 | ||
![]() |
2fdcd40f00 | ||
![]() |
3b15b786ff | ||
![]() |
b212b30e58 | ||
![]() |
6fd89f8585 | ||
![]() |
0406d21703 | ||
![]() |
293f89a07b | ||
![]() |
520a0b4075 | ||
![]() |
488602e232 | ||
![]() |
1e8d353162 | ||
![]() |
b3718b8b4a | ||
![]() |
097cba5c60 | ||
![]() |
fa6d8d0891 | ||
![]() |
31797c55df | ||
![]() |
56a23c5c3d | ||
![]() |
adc89f1487 | ||
![]() |
7facc375bc | ||
![]() |
91d3fb0ea8 | ||
![]() |
4ab0047dc1 | ||
![]() |
279eeaa442 | ||
![]() |
d4d0fb2a03 | ||
![]() |
d56fe8a542 | ||
![]() |
292701925d | ||
![]() |
52fc854cc3 | ||
![]() |
90ca039768 | ||
![]() |
9e81055070 | ||
![]() |
cea402ebf8 | ||
![]() |
6b939b95c0 | ||
![]() |
b24621d1ea | ||
![]() |
cc0fde2c08 | ||
![]() |
3732998fb7 | ||
![]() |
c699e265ef | ||
![]() |
98bb726f1a | ||
![]() |
c132e7ed85 | ||
![]() |
298cebe17f | ||
![]() |
d03825d200 | ||
![]() |
d9ab9db211 | ||
![]() |
58a607561a | ||
![]() |
0ae1f11ffc | ||
![]() |
db48c5a6a3 | ||
![]() |
effefdbff1 | ||
![]() |
233c969402 | ||
![]() |
3b885dd01f | ||
![]() |
52c8554d89 | ||
![]() |
b55baef985 | ||
![]() |
b593b15f27 | ||
![]() |
00669ac0c3 | ||
![]() |
33a4258c06 | ||
![]() |
d4a8fcbe03 | ||
![]() |
36c3b938ce | ||
![]() |
bf2fad2a2a | ||
![]() |
9cbd49b867 | ||
![]() |
9de59131f4 | ||
![]() |
7f44e89829 | ||
![]() |
572e4457b3 | ||
![]() |
a5bcf87c08 | ||
![]() |
8ca5b7528b | ||
![]() |
d951e68c10 | ||
![]() |
32e8d2043c | ||
![]() |
bf028915ec | ||
![]() |
b03f483e4f | ||
![]() |
6f6202eb69 | ||
![]() |
7ab2d1496e | ||
![]() |
acc229a7e1 | ||
![]() |
64ffa86fe3 | ||
![]() |
8b77024fb9 | ||
![]() |
42aa18ac16 | ||
![]() |
54d21666d0 | ||
![]() |
aac00a5e78 | ||
![]() |
63d93f2a36 | ||
![]() |
a9f453ea36 | ||
![]() |
d248de92e5 | ||
![]() |
0ed483ba51 | ||
![]() |
68fbadf21b | ||
![]() |
ac66079d41 | ||
![]() |
56c681bcf8 | ||
![]() |
c5c4253760 | ||
![]() |
84e6f2fc4f | ||
![]() |
8cedaae645 | ||
![]() |
e350ba4726 | ||
![]() |
1b7742ef7f | ||
![]() |
0c6bf701c7 | ||
![]() |
05e2e305e4 | ||
![]() |
5523cd6203 | ||
![]() |
50da4bcd37 | ||
![]() |
b99072d986 | ||
![]() |
b9a7a7c422 | ||
![]() |
88ccbcd883 | ||
![]() |
b5bb6c6fe5 | ||
![]() |
19a3810168 | ||
![]() |
8ccc38eb00 | ||
![]() |
70146a08c1 | ||
![]() |
19d50b9c92 | ||
![]() |
05c1328ca7 | ||
![]() |
99c2dd9765 | ||
![]() |
edbe6851f7 | ||
![]() |
a7867a9253 | ||
![]() |
94e70f81ed | ||
![]() |
3d8654253a | ||
![]() |
69dbcec678 | ||
![]() |
de8b0ba8c5 | ||
![]() |
730cd9f983 | ||
![]() |
67d8765624 | ||
![]() |
39bd07de73 | ||
![]() |
3202ea55d2 | ||
![]() |
329a8c0c90 | ||
![]() |
c05824c641 | ||
![]() |
3abdffda9c | ||
![]() |
67da851efc | ||
![]() |
5463a27255 | ||
![]() |
ec0434c9b0 | ||
![]() |
7d8cb5c863 | ||
![]() |
4f01348ffb | ||
![]() |
ca7e257e95 | ||
![]() |
a34332b48d | ||
![]() |
962912c43c | ||
![]() |
2af3400464 | ||
![]() |
b6e220a4c5 | ||
![]() |
d5d45f100e | ||
![]() |
6b9ca60c47 | ||
![]() |
bc445a1e27 | ||
![]() |
a087b4c43e | ||
![]() |
8f67ddf968 | ||
![]() |
9ef07484dd | ||
![]() |
7475cb56a1 | ||
![]() |
5287061699 | ||
![]() |
3ef1110109 | ||
![]() |
2efe2589d2 | ||
![]() |
4fb596357d | ||
![]() |
dd98ec771d | ||
![]() |
94f74308d8 | ||
![]() |
b982884933 | ||
![]() |
c47c6e358b | ||
![]() |
46394d0bf9 | ||
![]() |
4dc154201a | ||
![]() |
ebdbab81d3 | ||
![]() |
155098bc41 | ||
![]() |
291638a9dd | ||
![]() |
763c672e36 | ||
![]() |
c945534640 | ||
![]() |
5b3074d939 | ||
![]() |
3b89b72568 | ||
![]() |
1d9fa1522c | ||
![]() |
4db743db00 | ||
![]() |
d5f8231f97 | ||
![]() |
32c403d069 | ||
![]() |
220da51606 | ||
![]() |
9ae234a02f | ||
![]() |
401bbed67b | ||
![]() |
83190c21db | ||
![]() |
ccdd906e2f | ||
![]() |
f4c932ef9c | ||
![]() |
0892ed18e5 | ||
![]() |
3afc218adc | ||
![]() |
841b9c0917 | ||
![]() |
a479c6e786 | ||
![]() |
babb723521 | ||
![]() |
29954e530e | ||
![]() |
fb3c94f403 | ||
![]() |
dd8c1d359c | ||
![]() |
45e09a262b | ||
![]() |
d6d61a4137 | ||
![]() |
8fe7711634 | ||
![]() |
a5ec7fc251 | ||
![]() |
b9935717dc | ||
![]() |
bb25817bae | ||
![]() |
bf8a33e086 | ||
![]() |
bf56f50e0a | ||
![]() |
3a8e2c429f | ||
![]() |
e8fca5d93c | ||
![]() |
a39cf99024 | ||
![]() |
0ff27154e6 | ||
![]() |
766fd4cbf5 | ||
![]() |
1869260868 | ||
![]() |
93046d78f6 | ||
![]() |
3e51f9a505 | ||
![]() |
d95bf64edf | ||
![]() |
47f7cf5419 | ||
![]() |
267fc3743d | ||
![]() |
a6d73f7615 | ||
![]() |
b360c854a8 | ||
![]() |
a088b20987 | ||
![]() |
af6dd545dc | ||
![]() |
a26df88022 | ||
![]() |
86ec272581 | ||
![]() |
2a803e09a4 | ||
![]() |
35ebfc15c9 | ||
![]() |
f0a9185e4a | ||
![]() |
b3766cbc62 | ||
![]() |
3469013f1a | ||
![]() |
03486e4125 | ||
![]() |
20560fb847 | ||
![]() |
bad18da658 | ||
![]() |
e26c7c491a | ||
![]() |
b0c8ae0c94 | ||
![]() |
cc1658cbab | ||
![]() |
4b768f0635 | ||
![]() |
989057d947 | ||
![]() |
6671d24fa6 | ||
![]() |
297c721229 | ||
![]() |
70c502bb45 | ||
![]() |
add0b55657 | ||
![]() |
50e559487d | ||
![]() |
4cd02c81bb | ||
![]() |
d044f4d34e | ||
![]() |
696717dd90 | ||
![]() |
22c3132638 | ||
![]() |
ce32de6e23 |
@@ -1,5 +1,5 @@
|
|||||||
# See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile
|
# See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile
|
||||||
FROM mcr.microsoft.com/vscode/devcontainers/python:0-3.11
|
FROM mcr.microsoft.com/devcontainers/python:3.12
|
||||||
|
|
||||||
ENV \
|
ENV \
|
||||||
DEBIAN_FRONTEND=noninteractive \
|
DEBIAN_FRONTEND=noninteractive \
|
||||||
|
4
.github/workflows/cast_deployment.yaml
vendored
4
.github/workflows/cast_deployment.yaml
vendored
@@ -21,7 +21,7 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
with:
|
with:
|
||||||
ref: dev
|
ref: dev
|
||||||
|
|
||||||
@@ -57,7 +57,7 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
with:
|
with:
|
||||||
ref: master
|
ref: master
|
||||||
|
|
||||||
|
14
.github/workflows/ci.yaml
vendored
14
.github/workflows/ci.yaml
vendored
@@ -24,7 +24,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v4.0.2
|
||||||
with:
|
with:
|
||||||
@@ -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.0.0
|
uses: actions/cache@v4.0.2
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
node_modules/.cache/prettier
|
node_modules/.cache/prettier
|
||||||
@@ -58,7 +58,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v4.0.2
|
||||||
with:
|
with:
|
||||||
@@ -76,7 +76,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v4.0.2
|
||||||
with:
|
with:
|
||||||
@@ -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.3.1
|
uses: actions/upload-artifact@v4.3.2
|
||||||
with:
|
with:
|
||||||
name: frontend-bundle-stats
|
name: frontend-bundle-stats
|
||||||
path: build/stats/*.json
|
path: build/stats/*.json
|
||||||
@@ -100,7 +100,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v4.0.2
|
||||||
with:
|
with:
|
||||||
@@ -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.3.1
|
uses: actions/upload-artifact@v4.3.2
|
||||||
with:
|
with:
|
||||||
name: supervisor-bundle-stats
|
name: supervisor-bundle-stats
|
||||||
path: build/stats/*.json
|
path: build/stats/*.json
|
||||||
|
2
.github/workflows/codeql-analysis.yml
vendored
2
.github/workflows/codeql-analysis.yml
vendored
@@ -23,7 +23,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
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.
|
||||||
|
4
.github/workflows/demo_deployment.yaml
vendored
4
.github/workflows/demo_deployment.yaml
vendored
@@ -22,7 +22,7 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
with:
|
with:
|
||||||
ref: dev
|
ref: dev
|
||||||
|
|
||||||
@@ -58,7 +58,7 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
with:
|
with:
|
||||||
ref: master
|
ref: master
|
||||||
|
|
||||||
|
2
.github/workflows/design_deployment.yaml
vendored
2
.github/workflows/design_deployment.yaml
vendored
@@ -16,7 +16,7 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v4.0.2
|
||||||
|
2
.github/workflows/design_preview.yaml
vendored
2
.github/workflows/design_preview.yaml
vendored
@@ -21,7 +21,7 @@ jobs:
|
|||||||
if: github.repository == 'home-assistant/frontend' && contains(github.event.pull_request.labels.*.name, 'needs design preview')
|
if: github.repository == 'home-assistant/frontend' && contains(github.event.pull_request.labels.*.name, 'needs design preview')
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v4.0.2
|
||||||
|
8
.github/workflows/nightly.yaml
vendored
8
.github/workflows/nightly.yaml
vendored
@@ -6,7 +6,7 @@ on:
|
|||||||
- cron: "0 1 * * *"
|
- cron: "0 1 * * *"
|
||||||
|
|
||||||
env:
|
env:
|
||||||
PYTHON_VERSION: "3.11"
|
PYTHON_VERSION: "3.12"
|
||||||
NODE_OPTIONS: --max_old_space_size=6144
|
NODE_OPTIONS: --max_old_space_size=6144
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
@@ -20,7 +20,7 @@ jobs:
|
|||||||
contents: write
|
contents: write
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
|
|
||||||
- name: Set up Python ${{ env.PYTHON_VERSION }}
|
- name: Set up Python ${{ env.PYTHON_VERSION }}
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v5
|
||||||
@@ -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.3.1
|
uses: actions/upload-artifact@v4.3.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.3.1
|
uses: actions/upload-artifact@v4.3.2
|
||||||
with:
|
with:
|
||||||
name: translations
|
name: translations
|
||||||
path: translations.tar.gz
|
path: translations.tar.gz
|
||||||
|
6
.github/workflows/release.yaml
vendored
6
.github/workflows/release.yaml
vendored
@@ -6,7 +6,7 @@ on:
|
|||||||
- published
|
- published
|
||||||
|
|
||||||
env:
|
env:
|
||||||
PYTHON_VERSION: "3.11"
|
PYTHON_VERSION: "3.12"
|
||||||
NODE_OPTIONS: --max_old_space_size=6144
|
NODE_OPTIONS: --max_old_space_size=6144
|
||||||
|
|
||||||
# Set default workflow permissions
|
# Set default workflow permissions
|
||||||
@@ -23,7 +23,7 @@ jobs:
|
|||||||
contents: write # Required to upload release assets
|
contents: write # Required to upload release assets
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
|
|
||||||
- name: Verify version
|
- name: Verify version
|
||||||
uses: home-assistant/actions/helpers/verify-version@master
|
uses: home-assistant/actions/helpers/verify-version@master
|
||||||
@@ -55,7 +55,7 @@ jobs:
|
|||||||
script/release
|
script/release
|
||||||
|
|
||||||
- name: Upload release assets
|
- name: Upload release assets
|
||||||
uses: softprops/action-gh-release@v0.1.15
|
uses: softprops/action-gh-release@v2.0.4
|
||||||
with:
|
with:
|
||||||
files: |
|
files: |
|
||||||
dist/*.whl
|
dist/*.whl
|
||||||
|
2
.github/workflows/translations.yaml
vendored
2
.github/workflows/translations.yaml
vendored
@@ -13,7 +13,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@v4.1.1
|
uses: actions/checkout@v4.1.3
|
||||||
|
|
||||||
- name: Upload Translations
|
- name: Upload Translations
|
||||||
run: |
|
run: |
|
||||||
|
@@ -1,13 +0,0 @@
|
|||||||
diff --git a/simple-tooltip.js b/simple-tooltip.js
|
|
||||||
index 78a87f6a223925f0e29fbedb268c85a142ec6985..3d686dd6a3d5a93342b4b01408089fc316b408ca 100644
|
|
||||||
--- a/simple-tooltip.js
|
|
||||||
+++ b/simple-tooltip.js
|
|
||||||
@@ -195,6 +195,8 @@ class SimpleTooltip extends LitElement {
|
|
||||||
.hidden {
|
|
||||||
position: absolute;
|
|
||||||
left: -10000px;
|
|
||||||
+ inset-inline-start: -10000px;
|
|
||||||
+ inset-inline-end: initial;
|
|
||||||
top: auto;
|
|
||||||
width: 1px;
|
|
||||||
height: 1px;
|
|
18
.yarn/patches/hls.js-npm-1.5.7-f5bbd3d060.patch
Normal file
18
.yarn/patches/hls.js-npm-1.5.7-f5bbd3d060.patch
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
diff --git a/dist/hls.light.mjs b/dist/hls.light.mjs
|
||||||
|
index eed9d788fafdb159975e1a2eb08ac88ba9c9ac33..ace881935e6665946f1c8110ebd2f739cde4427e 100644
|
||||||
|
--- a/dist/hls.light.mjs
|
||||||
|
+++ b/dist/hls.light.mjs
|
||||||
|
@@ -20523,9 +20523,9 @@ class Hls {
|
||||||
|
}
|
||||||
|
Hls.defaultConfig = void 0;
|
||||||
|
|
||||||
|
-var KeySystemFormats = empty.KeySystemFormats;
|
||||||
|
-var KeySystems = empty.KeySystems;
|
||||||
|
-var SubtitleStreamController = empty.SubtitleStreamController;
|
||||||
|
-var TimelineController = empty.TimelineController;
|
||||||
|
+var KeySystemFormats = empty;
|
||||||
|
+var KeySystems = empty;
|
||||||
|
+var SubtitleStreamController = empty;
|
||||||
|
+var TimelineController = empty;
|
||||||
|
export { AbrController, AttrList, Cues as AudioStreamController, Cues as AudioTrackController, BasePlaylistController, BaseSegment, BaseStreamController, BufferController, Cues as CMCDController, CapLevelController, ChunkMetadata, ContentSteeringController, DateRange, Cues as EMEController, ErrorActionFlags, ErrorController, ErrorDetails, ErrorTypes, Events, FPSController, Fragment, Hls, HlsSkip, HlsUrlParameters, KeySystemFormats, KeySystems, Level, LevelDetails, LevelKey, LoadStats, MetadataSchema, NetworkErrorAction, Part, PlaylistLevelType, SubtitleStreamController, Cues as SubtitleTrackController, TimelineController, Hls as default, getMediaSource, isMSESupported, isSupported };
|
||||||
|
//# sourceMappingURL=hls.light.mjs.map
|
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.1.0.cjs
|
yarnPath: .yarn/releases/yarn-4.1.1.cjs
|
||||||
|
@@ -1,26 +1,19 @@
|
|||||||
// Tasks to compress
|
// Tasks to compress
|
||||||
|
|
||||||
import { deleteAsync } from "del";
|
|
||||||
import gulp from "gulp";
|
import gulp from "gulp";
|
||||||
import gulpIf from "gulp-if";
|
|
||||||
import vinylPaths from "vinyl-paths";
|
|
||||||
import zopfli from "gulp-zopfli-green";
|
import zopfli from "gulp-zopfli-green";
|
||||||
import paths from "../paths.cjs";
|
import paths from "../paths.cjs";
|
||||||
|
|
||||||
const zopfliOptions = { threshold: 150 };
|
const zopfliOptions = { threshold: 150 };
|
||||||
|
|
||||||
const compressedExt = /\.gz$/;
|
|
||||||
const deleteUncompressed = (p) => deleteAsync(p.replace(compressedExt, ""));
|
|
||||||
|
|
||||||
const compressDist = (rootDir) =>
|
const compressDist = (rootDir) =>
|
||||||
gulp
|
gulp
|
||||||
.src([
|
.src([
|
||||||
`${rootDir}/**/*.{js?(.map),json,css,svg,xml}`,
|
`${rootDir}/**/*.{js,json,css,svg,xml}`,
|
||||||
`${rootDir}/{authorize,onboarding}.html`,
|
`${rootDir}/{authorize,onboarding}.html`,
|
||||||
])
|
])
|
||||||
.pipe(zopfli(zopfliOptions))
|
.pipe(zopfli(zopfliOptions))
|
||||||
.pipe(gulp.dest(rootDir))
|
.pipe(gulp.dest(rootDir));
|
||||||
.pipe(gulpIf(compressedExt, vinylPaths(deleteUncompressed)));
|
|
||||||
|
|
||||||
gulp.task("compress-app", () => compressDist(paths.app_output_root));
|
gulp.task("compress-app", () => compressDist(paths.app_output_root));
|
||||||
gulp.task("compress-hassio", () => compressDist(paths.hassio_output_root));
|
gulp.task("compress-hassio", () => compressDist(paths.hassio_output_root));
|
||||||
|
@@ -9,7 +9,7 @@ import gulp from "gulp";
|
|||||||
import jszip from "jszip";
|
import jszip from "jszip";
|
||||||
import path from "path";
|
import path from "path";
|
||||||
import process from "process";
|
import process from "process";
|
||||||
import tar from "tar";
|
import { extract } from "tar";
|
||||||
|
|
||||||
const MAX_AGE = 24; // hours
|
const MAX_AGE = 24; // hours
|
||||||
const OWNER = "home-assistant";
|
const OWNER = "home-assistant";
|
||||||
@@ -156,7 +156,7 @@ gulp.task("fetch-nightly-translations", async function () {
|
|||||||
console.log("Unpacking downloaded translations...");
|
console.log("Unpacking downloaded translations...");
|
||||||
const zip = await jszip.loadAsync(downloadResponse.data);
|
const zip = await jszip.loadAsync(downloadResponse.data);
|
||||||
await deleteCurrent;
|
await deleteCurrent;
|
||||||
const extractStream = zip.file(/.*/)[0].nodeStream().pipe(tar.extract());
|
const extractStream = zip.file(/.*/)[0].nodeStream().pipe(extract());
|
||||||
await new Promise((resolve, reject) => {
|
await new Promise((resolve, reject) => {
|
||||||
extractStream.on("close", resolve).on("error", reject);
|
extractStream.on("close", resolve).on("error", reject);
|
||||||
});
|
});
|
||||||
|
@@ -1,92 +1,76 @@
|
|||||||
import { createHash } from "crypto";
|
import { deleteAsync } from "del";
|
||||||
import { deleteSync } from "del";
|
import { glob } from "glob";
|
||||||
import { mkdirSync, readdirSync, readFileSync, renameSync } from "fs";
|
|
||||||
import { writeFile } from "node:fs/promises";
|
|
||||||
import gulp from "gulp";
|
import gulp from "gulp";
|
||||||
import flatmap from "gulp-flatmap";
|
|
||||||
import transform from "gulp-json-transform";
|
|
||||||
import merge from "gulp-merge-json";
|
import merge from "gulp-merge-json";
|
||||||
import rename from "gulp-rename";
|
import rename from "gulp-rename";
|
||||||
import path from "path";
|
import { createHash } from "node:crypto";
|
||||||
import vinylBuffer from "vinyl-buffer";
|
import { mkdir, readFile } from "node:fs/promises";
|
||||||
import source from "vinyl-source-stream";
|
import { basename, join } from "node:path";
|
||||||
|
import { Transform } from "node:stream";
|
||||||
|
import { finished } from "node:stream/promises";
|
||||||
import env from "../env.cjs";
|
import env from "../env.cjs";
|
||||||
import paths from "../paths.cjs";
|
import paths from "../paths.cjs";
|
||||||
import { mapFiles } from "../util.cjs";
|
|
||||||
import "./fetch-nightly-translations.js";
|
import "./fetch-nightly-translations.js";
|
||||||
|
|
||||||
const inFrontendDir = "translations/frontend";
|
const inFrontendDir = "translations/frontend";
|
||||||
const inBackendDir = "translations/backend";
|
const inBackendDir = "translations/backend";
|
||||||
const workDir = "build/translations";
|
const workDir = "build/translations";
|
||||||
const fullDir = workDir + "/full";
|
const outDir = join(workDir, "output");
|
||||||
const coreDir = workDir + "/core";
|
const EN_SRC = join(paths.translations_src, "en.json");
|
||||||
const outDir = workDir + "/output";
|
|
||||||
let mergeBackend = false;
|
let mergeBackend = false;
|
||||||
|
|
||||||
gulp.task(
|
gulp.task(
|
||||||
"translations-enable-merge-backend",
|
"translations-enable-merge-backend",
|
||||||
gulp.parallel((done) => {
|
gulp.parallel(async () => {
|
||||||
mergeBackend = true;
|
mergeBackend = true;
|
||||||
done();
|
|
||||||
}, "allow-setup-fetch-nightly-translations")
|
}, "allow-setup-fetch-nightly-translations")
|
||||||
);
|
);
|
||||||
|
|
||||||
// Panel translations which should be split from the core translations.
|
// Transform stream to apply a function on Vinyl JSON files (buffer mode only).
|
||||||
const TRANSLATION_FRAGMENTS = Object.keys(
|
// The provided function can either return a new object, or an array of
|
||||||
JSON.parse(
|
// [object, subdirectory] pairs for fragmentizing the JSON.
|
||||||
readFileSync(
|
class CustomJSON extends Transform {
|
||||||
path.resolve(paths.polymer_dir, "src/translations/en.json"),
|
constructor(func, reviver = null) {
|
||||||
"utf-8"
|
super({ objectMode: true });
|
||||||
)
|
this._func = func;
|
||||||
).ui.panel
|
this._reviver = reviver;
|
||||||
);
|
}
|
||||||
|
|
||||||
function recursiveFlatten(prefix, data) {
|
async _transform(file, _, callback) {
|
||||||
let output = {};
|
try {
|
||||||
Object.keys(data).forEach((key) => {
|
let obj = JSON.parse(file.contents.toString(), this._reviver);
|
||||||
if (typeof data[key] === "object") {
|
if (this._func) obj = this._func(obj, file.path);
|
||||||
output = {
|
for (const [outObj, dir] of Array.isArray(obj) ? obj : [[obj, ""]]) {
|
||||||
...output,
|
const outFile = file.clone({ contents: false });
|
||||||
...recursiveFlatten(prefix + key + ".", data[key]),
|
outFile.contents = Buffer.from(JSON.stringify(outObj));
|
||||||
};
|
outFile.dirname += `/${dir}`;
|
||||||
|
this.push(outFile);
|
||||||
|
}
|
||||||
|
callback(null);
|
||||||
|
} catch (err) {
|
||||||
|
callback(err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Utility to flatten object keys to single level using separator
|
||||||
|
const flatten = (data, prefix = "", sep = ".") => {
|
||||||
|
const output = {};
|
||||||
|
for (const [key, value] of Object.entries(data)) {
|
||||||
|
if (typeof value === "object") {
|
||||||
|
Object.assign(output, flatten(value, prefix + key + sep, sep));
|
||||||
} else {
|
} else {
|
||||||
output[prefix + key] = data[key];
|
output[prefix + key] = value;
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
return output;
|
return output;
|
||||||
}
|
};
|
||||||
|
|
||||||
function flatten(data) {
|
// Filter functions that can be passed directly to JSON.parse()
|
||||||
return recursiveFlatten("", data);
|
const emptyReviver = (_key, value) => value || undefined;
|
||||||
}
|
const testReviver = (_key, value) =>
|
||||||
|
value && typeof value === "string" ? "TRANSLATED" : value;
|
||||||
function emptyFilter(data) {
|
|
||||||
const newData = {};
|
|
||||||
Object.keys(data).forEach((key) => {
|
|
||||||
if (data[key]) {
|
|
||||||
if (typeof data[key] === "object") {
|
|
||||||
newData[key] = emptyFilter(data[key]);
|
|
||||||
} else {
|
|
||||||
newData[key] = data[key];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return newData;
|
|
||||||
}
|
|
||||||
|
|
||||||
function recursiveEmpty(data) {
|
|
||||||
const newData = {};
|
|
||||||
Object.keys(data).forEach((key) => {
|
|
||||||
if (data[key]) {
|
|
||||||
if (typeof data[key] === "object") {
|
|
||||||
newData[key] = recursiveEmpty(data[key]);
|
|
||||||
} else {
|
|
||||||
newData[key] = "TRANSLATED";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return newData;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Replace Lokalise key placeholders with their actual values.
|
* Replace Lokalise key placeholders with their actual values.
|
||||||
@@ -95,60 +79,44 @@ function recursiveEmpty(data) {
|
|||||||
* be included in src/translations/en.json, but still be usable while
|
* be included in src/translations/en.json, but still be usable while
|
||||||
* developing locally.
|
* developing locally.
|
||||||
*
|
*
|
||||||
* @link https://docs.lokalise.co/article/KO5SZWLLsy-key-referencing
|
* @link https://docs.lokalise.com/en/articles/1400528-key-referencing
|
||||||
*/
|
*/
|
||||||
const re_key_reference = /\[%key:([^%]+)%\]/;
|
const KEY_REFERENCE = /\[%key:([^%]+)%\]/;
|
||||||
function lokaliseTransform(data, original, file) {
|
const lokaliseTransform = (data, path, original = data) => {
|
||||||
const output = {};
|
const output = {};
|
||||||
Object.entries(data).forEach(([key, value]) => {
|
for (const [key, value] of Object.entries(data)) {
|
||||||
if (value instanceof Object) {
|
if (typeof value === "object") {
|
||||||
output[key] = lokaliseTransform(value, original, file);
|
output[key] = lokaliseTransform(value, path, original);
|
||||||
} else {
|
} else {
|
||||||
output[key] = value.replace(re_key_reference, (_match, lokalise_key) => {
|
output[key] = value.replace(KEY_REFERENCE, (_match, lokalise_key) => {
|
||||||
const replace = lokalise_key.split("::").reduce((tr, k) => {
|
const replace = lokalise_key.split("::").reduce((tr, k) => {
|
||||||
if (!tr) {
|
if (!tr) {
|
||||||
throw Error(
|
throw Error(`Invalid key placeholder ${lokalise_key} in ${path}`);
|
||||||
`Invalid key placeholder ${lokalise_key} in ${file.path}`
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
return tr[k];
|
return tr[k];
|
||||||
}, original);
|
}, original);
|
||||||
if (typeof replace !== "string") {
|
if (typeof replace !== "string") {
|
||||||
throw Error(
|
throw Error(`Invalid key placeholder ${lokalise_key} in ${path}`);
|
||||||
`Invalid key placeholder ${lokalise_key} in ${file.path}`
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
return replace;
|
return replace;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
return output;
|
return output;
|
||||||
}
|
};
|
||||||
|
|
||||||
gulp.task("clean-translations", async () => deleteSync([workDir]));
|
gulp.task("clean-translations", () => deleteAsync([workDir]));
|
||||||
|
|
||||||
gulp.task("ensure-translations-build-dir", async () => {
|
const makeWorkDir = () => mkdir(workDir, { recursive: true });
|
||||||
mkdirSync(workDir, { recursive: true });
|
|
||||||
});
|
|
||||||
|
|
||||||
gulp.task("create-test-metadata", () =>
|
const createTestTranslation = () =>
|
||||||
env.isProdBuild()
|
|
||||||
? Promise.resolve()
|
|
||||||
: writeFile(
|
|
||||||
workDir + "/testMetadata.json",
|
|
||||||
JSON.stringify({ test: { nativeName: "Test" } })
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
gulp.task("create-test-translation", () =>
|
|
||||||
env.isProdBuild()
|
env.isProdBuild()
|
||||||
? Promise.resolve()
|
? Promise.resolve()
|
||||||
: gulp
|
: gulp
|
||||||
.src(path.join(paths.translations_src, "en.json"))
|
.src(EN_SRC)
|
||||||
.pipe(transform((data, _file) => recursiveEmpty(data)))
|
.pipe(new CustomJSON(null, testReviver))
|
||||||
.pipe(rename("test.json"))
|
.pipe(rename("test.json"))
|
||||||
.pipe(gulp.dest(workDir))
|
.pipe(gulp.dest(workDir));
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This task will build a master translation file, to be used as the base for
|
* This task will build a master translation file, to be used as the base for
|
||||||
@@ -159,279 +127,171 @@ gulp.task("create-test-translation", () =>
|
|||||||
* project is buildable immediately after merging new translation keys, since
|
* project is buildable immediately after merging new translation keys, since
|
||||||
* the Lokalise update to translations/en.json will not happen immediately.
|
* the Lokalise update to translations/en.json will not happen immediately.
|
||||||
*/
|
*/
|
||||||
gulp.task("build-master-translation", () => {
|
const createMasterTranslation = () =>
|
||||||
const src = [path.join(paths.translations_src, "en.json")];
|
gulp
|
||||||
|
.src([EN_SRC, ...(mergeBackend ? [`${inBackendDir}/en.json`] : [])])
|
||||||
if (mergeBackend) {
|
.pipe(new CustomJSON(lokaliseTransform))
|
||||||
src.push(path.join(inBackendDir, "en.json"));
|
|
||||||
}
|
|
||||||
|
|
||||||
return gulp
|
|
||||||
.src(src)
|
|
||||||
.pipe(transform((data, file) => lokaliseTransform(data, data, file)))
|
|
||||||
.pipe(
|
.pipe(
|
||||||
merge({
|
merge({
|
||||||
fileName: "en.json",
|
fileName: "en.json",
|
||||||
|
jsonSpace: undefined,
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
.pipe(gulp.dest(fullDir));
|
.pipe(gulp.dest(workDir));
|
||||||
});
|
|
||||||
|
|
||||||
gulp.task("build-merged-translations", () =>
|
const FRAGMENTS = ["base"];
|
||||||
gulp
|
|
||||||
.src([
|
|
||||||
inFrontendDir + "/*.json",
|
|
||||||
"!" + inFrontendDir + "/en.json",
|
|
||||||
...(env.isProdBuild() ? [] : [workDir + "/test.json"]),
|
|
||||||
])
|
|
||||||
.pipe(transform((data, file) => lokaliseTransform(data, data, file)))
|
|
||||||
.pipe(
|
|
||||||
flatmap((stream, file) => {
|
|
||||||
// For each language generate a merged json file. It begins with the master
|
|
||||||
// translation as a failsafe for untranslated strings, and merges all parent
|
|
||||||
// tags into one file for each specific subtag
|
|
||||||
//
|
|
||||||
// TODO: This is a naive interpretation of BCP47 that should be improved.
|
|
||||||
// Will be OK for now as long as we don't have anything more complicated
|
|
||||||
// than a base translation + region.
|
|
||||||
const tr = path.basename(file.history[0], ".json");
|
|
||||||
const subtags = tr.split("-");
|
|
||||||
const src = [fullDir + "/en.json"];
|
|
||||||
for (let i = 1; i <= subtags.length; i++) {
|
|
||||||
const lang = subtags.slice(0, i).join("-");
|
|
||||||
if (lang === "test") {
|
|
||||||
src.push(workDir + "/test.json");
|
|
||||||
} else if (lang !== "en") {
|
|
||||||
src.push(inFrontendDir + "/" + lang + ".json");
|
|
||||||
if (mergeBackend) {
|
|
||||||
src.push(inBackendDir + "/" + lang + ".json");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return gulp
|
|
||||||
.src(src, { allowEmpty: true })
|
|
||||||
.pipe(transform((data) => emptyFilter(data)))
|
|
||||||
.pipe(
|
|
||||||
merge({
|
|
||||||
fileName: tr + ".json",
|
|
||||||
})
|
|
||||||
)
|
|
||||||
.pipe(gulp.dest(fullDir));
|
|
||||||
})
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
let taskName;
|
const toggleSupervisorFragment = async () => {
|
||||||
|
FRAGMENTS[0] = "supervisor";
|
||||||
|
};
|
||||||
|
|
||||||
const splitTasks = [];
|
const panelFragment = (fragment) =>
|
||||||
TRANSLATION_FRAGMENTS.forEach((fragment) => {
|
fragment !== "base" && fragment !== "supervisor";
|
||||||
taskName = "build-translation-fragment-" + fragment;
|
|
||||||
gulp.task(taskName, () =>
|
|
||||||
// Return only the translations for this fragment.
|
|
||||||
gulp
|
|
||||||
.src(fullDir + "/*.json")
|
|
||||||
.pipe(
|
|
||||||
transform((data) => ({
|
|
||||||
ui: {
|
|
||||||
panel: {
|
|
||||||
[fragment]: data.ui.panel[fragment],
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}))
|
|
||||||
)
|
|
||||||
.pipe(gulp.dest(workDir + "/" + fragment))
|
|
||||||
);
|
|
||||||
splitTasks.push(taskName);
|
|
||||||
});
|
|
||||||
|
|
||||||
taskName = "build-translation-core";
|
const HASHES = new Map();
|
||||||
gulp.task(taskName, () =>
|
|
||||||
// Remove the fragment translations from the core translation.
|
|
||||||
gulp
|
|
||||||
.src(fullDir + "/*.json")
|
|
||||||
.pipe(
|
|
||||||
transform((data, _file) => {
|
|
||||||
TRANSLATION_FRAGMENTS.forEach((fragment) => {
|
|
||||||
delete data.ui.panel[fragment];
|
|
||||||
});
|
|
||||||
delete data.supervisor;
|
|
||||||
return data;
|
|
||||||
})
|
|
||||||
)
|
|
||||||
.pipe(gulp.dest(coreDir))
|
|
||||||
);
|
|
||||||
|
|
||||||
splitTasks.push(taskName);
|
const createTranslations = async () => {
|
||||||
|
// Parse and store the master to avoid repeating this for each locale, then
|
||||||
gulp.task("build-flattened-translations", () =>
|
// add the panel fragments when processing the app.
|
||||||
// Flatten the split versions of our translations, and move them into outDir
|
const enMaster = JSON.parse(await readFile(`${workDir}/en.json`, "utf-8"));
|
||||||
gulp
|
if (FRAGMENTS[0] === "base") {
|
||||||
.src(
|
FRAGMENTS.push(...Object.keys(enMaster.ui.panel));
|
||||||
TRANSLATION_FRAGMENTS.map(
|
|
||||||
(fragment) => workDir + "/" + fragment + "/*.json"
|
|
||||||
).concat(coreDir + "/*.json"),
|
|
||||||
{ base: workDir }
|
|
||||||
)
|
|
||||||
.pipe(
|
|
||||||
transform((data) =>
|
|
||||||
// Polymer.AppLocalizeBehavior requires flattened json
|
|
||||||
flatten(data)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
.pipe(
|
|
||||||
rename((filePath) => {
|
|
||||||
if (filePath.dirname === "core") {
|
|
||||||
filePath.dirname = "";
|
|
||||||
}
|
|
||||||
// In dev we create the file with the fake hash in the filename
|
|
||||||
if (!env.isProdBuild()) {
|
|
||||||
filePath.basename += "-dev";
|
|
||||||
}
|
|
||||||
})
|
|
||||||
)
|
|
||||||
.pipe(gulp.dest(outDir))
|
|
||||||
);
|
|
||||||
|
|
||||||
const fingerprints = {};
|
|
||||||
|
|
||||||
gulp.task("build-translation-fingerprints", () => {
|
|
||||||
// Fingerprint full file of each language
|
|
||||||
const files = readdirSync(fullDir);
|
|
||||||
|
|
||||||
for (let i = 0; i < files.length; i++) {
|
|
||||||
fingerprints[files[i].split(".")[0]] = {
|
|
||||||
// In dev we create fake hashes
|
|
||||||
hash: env.isProdBuild()
|
|
||||||
? createHash("md5")
|
|
||||||
.update(readFileSync(path.join(fullDir, files[i]), "utf-8"))
|
|
||||||
.digest("hex")
|
|
||||||
: "dev",
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// In dev we create the file with the fake hash in the filename
|
// The downstream pipeline is setup first. It hashes the merged data for
|
||||||
if (env.isProdBuild()) {
|
// each locale, then fragmentizes and flattens the data for final output.
|
||||||
mapFiles(outDir, ".json", (filename) => {
|
const translationFiles = await glob([
|
||||||
const parsed = path.parse(filename);
|
`${inFrontendDir}/!(en).json`,
|
||||||
|
...(env.isProdBuild() ? [] : [`${workDir}/test.json`]),
|
||||||
|
]);
|
||||||
|
const hashStream = new Transform({
|
||||||
|
objectMode: true,
|
||||||
|
transform: async (file, _, callback) => {
|
||||||
|
const hash = env.isProdBuild()
|
||||||
|
? createHash("md5").update(file.contents).digest("hex")
|
||||||
|
: "dev";
|
||||||
|
HASHES.set(file.stem, hash);
|
||||||
|
file.stem += `-${hash}`;
|
||||||
|
callback(null, file);
|
||||||
|
},
|
||||||
|
}).setMaxListeners(translationFiles.length + 1);
|
||||||
|
const fragmentsStream = hashStream
|
||||||
|
.pipe(
|
||||||
|
new CustomJSON((data) =>
|
||||||
|
FRAGMENTS.map((fragment) => {
|
||||||
|
switch (fragment) {
|
||||||
|
case "base":
|
||||||
|
// Remove the panels and supervisor to create the base translations
|
||||||
|
return [
|
||||||
|
flatten({
|
||||||
|
...data,
|
||||||
|
ui: { ...data.ui, panel: undefined },
|
||||||
|
supervisor: undefined,
|
||||||
|
}),
|
||||||
|
"",
|
||||||
|
];
|
||||||
|
case "supervisor":
|
||||||
|
// Supervisor key is at the top level
|
||||||
|
return [flatten(data.supervisor), ""];
|
||||||
|
default:
|
||||||
|
// Create a fragment with only the given panel
|
||||||
|
return [
|
||||||
|
flatten(data.ui.panel[fragment], `ui.panel.${fragment}.`),
|
||||||
|
fragment,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
})
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.pipe(gulp.dest(outDir));
|
||||||
|
|
||||||
// nl.json -> nl-<hash>.json
|
// Send the English master downstream first, then for each other locale
|
||||||
if (!(parsed.name in fingerprints)) {
|
// generate merged JSON data to continue piping. It begins with the master
|
||||||
throw new Error(`Unable to find hash for ${filename}`);
|
// translation as a failsafe for untranslated strings, and merges all parent
|
||||||
|
// tags into one file for each specific subtag
|
||||||
|
//
|
||||||
|
// TODO: This is a naive interpretation of BCP47 that should be improved.
|
||||||
|
// Will be OK for now as long as we don't have anything more complicated
|
||||||
|
// than a base translation + region.
|
||||||
|
gulp.src(`${workDir}/en.json`).pipe(hashStream, { end: false });
|
||||||
|
const mergesFinished = [];
|
||||||
|
for (const translationFile of translationFiles) {
|
||||||
|
const locale = basename(translationFile, ".json");
|
||||||
|
const subtags = locale.split("-");
|
||||||
|
const mergeFiles = [];
|
||||||
|
for (let i = 1; i <= subtags.length; i++) {
|
||||||
|
const lang = subtags.slice(0, i).join("-");
|
||||||
|
if (lang === "test") {
|
||||||
|
mergeFiles.push(`${workDir}/test.json`);
|
||||||
|
} else if (lang !== "en") {
|
||||||
|
mergeFiles.push(`${inFrontendDir}/${lang}.json`);
|
||||||
|
if (mergeBackend) {
|
||||||
|
mergeFiles.push(`${inBackendDir}/${lang}.json`);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
renameSync(
|
const mergeStream = gulp.src(mergeFiles, { allowEmpty: true }).pipe(
|
||||||
filename,
|
merge({
|
||||||
`${parsed.dir}/${parsed.name}-${fingerprints[parsed.name].hash}${
|
fileName: `${locale}.json`,
|
||||||
parsed.ext
|
startObj: enMaster,
|
||||||
}`
|
jsonReviver: emptyReviver,
|
||||||
);
|
jsonSpace: undefined,
|
||||||
});
|
})
|
||||||
|
);
|
||||||
|
mergesFinished.push(finished(mergeStream));
|
||||||
|
mergeStream.pipe(hashStream, { end: false });
|
||||||
}
|
}
|
||||||
|
|
||||||
const stream = source("translationFingerprints.json");
|
// Wait for all merges to finish, then it's safe to end writing to the
|
||||||
stream.write(JSON.stringify(fingerprints));
|
// downstream pipeline and wait for all fragments to finish writing.
|
||||||
process.nextTick(() => stream.end());
|
await Promise.all(mergesFinished);
|
||||||
return stream.pipe(vinylBuffer()).pipe(gulp.dest(workDir));
|
hashStream.end();
|
||||||
});
|
await finished(fragmentsStream);
|
||||||
|
};
|
||||||
|
|
||||||
gulp.task("build-translation-fragment-supervisor", () =>
|
const writeTranslationMetaData = () =>
|
||||||
gulp
|
gulp
|
||||||
.src(fullDir + "/*.json")
|
.src([`${paths.translations_src}/translationMetadata.json`])
|
||||||
.pipe(transform((data) => data.supervisor))
|
|
||||||
.pipe(
|
.pipe(
|
||||||
rename((filePath) => {
|
new CustomJSON((meta) => {
|
||||||
// In dev we create the file with the fake hash in the filename
|
// Add the test translation in development.
|
||||||
if (!env.isProdBuild()) {
|
if (!env.isProdBuild()) {
|
||||||
filePath.basename += "-dev";
|
meta.test = { nativeName: "Test" };
|
||||||
}
|
}
|
||||||
})
|
// Filter out locales without a native name, and add the hashes.
|
||||||
)
|
for (const locale of Object.keys(meta)) {
|
||||||
.pipe(gulp.dest(workDir + "/supervisor"))
|
if (!meta[locale].nativeName) {
|
||||||
);
|
meta[locale] = undefined;
|
||||||
|
|
||||||
gulp.task("build-translation-flatten-supervisor", () =>
|
|
||||||
gulp
|
|
||||||
.src(workDir + "/supervisor/*.json")
|
|
||||||
.pipe(
|
|
||||||
transform((data) =>
|
|
||||||
// Polymer.AppLocalizeBehavior requires flattened json
|
|
||||||
flatten(data)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
.pipe(gulp.dest(outDir))
|
|
||||||
);
|
|
||||||
|
|
||||||
gulp.task("build-translation-write-metadata", () =>
|
|
||||||
gulp
|
|
||||||
.src([
|
|
||||||
path.join(paths.translations_src, "translationMetadata.json"),
|
|
||||||
...(env.isProdBuild() ? [] : [workDir + "/testMetadata.json"]),
|
|
||||||
workDir + "/translationFingerprints.json",
|
|
||||||
])
|
|
||||||
.pipe(merge({}))
|
|
||||||
.pipe(
|
|
||||||
transform((data) => {
|
|
||||||
const newData = {};
|
|
||||||
Object.entries(data).forEach(([key, value]) => {
|
|
||||||
// Filter out translations without native name.
|
|
||||||
if (value.nativeName) {
|
|
||||||
newData[key] = value;
|
|
||||||
} else {
|
|
||||||
console.warn(
|
console.warn(
|
||||||
`Skipping language ${key}. Native name was not translated.`
|
`Skipping locale ${locale} because native name is not translated.`
|
||||||
);
|
);
|
||||||
|
} else {
|
||||||
|
meta[locale].hash = HASHES.get(locale);
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
return newData;
|
return {
|
||||||
|
fragments: FRAGMENTS.filter(panelFragment),
|
||||||
|
translations: meta,
|
||||||
|
};
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
.pipe(
|
.pipe(gulp.dest(workDir));
|
||||||
transform((data) => ({
|
|
||||||
fragments: TRANSLATION_FRAGMENTS,
|
|
||||||
translations: data,
|
|
||||||
}))
|
|
||||||
)
|
|
||||||
.pipe(rename("translationMetadata.json"))
|
|
||||||
.pipe(gulp.dest(workDir))
|
|
||||||
);
|
|
||||||
|
|
||||||
gulp.task(
|
|
||||||
"create-translations",
|
|
||||||
gulp.series(
|
|
||||||
gulp.parallel("create-test-metadata", "create-test-translation"),
|
|
||||||
"build-master-translation",
|
|
||||||
"build-merged-translations",
|
|
||||||
gulp.parallel(...splitTasks),
|
|
||||||
"build-flattened-translations"
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
gulp.task(
|
gulp.task(
|
||||||
"build-translations",
|
"build-translations",
|
||||||
gulp.series(
|
gulp.series(
|
||||||
gulp.parallel(
|
gulp.parallel(
|
||||||
"fetch-nightly-translations",
|
"fetch-nightly-translations",
|
||||||
gulp.series("clean-translations", "ensure-translations-build-dir")
|
gulp.series("clean-translations", makeWorkDir)
|
||||||
),
|
),
|
||||||
"create-translations",
|
createTestTranslation,
|
||||||
"build-translation-fingerprints",
|
createMasterTranslation,
|
||||||
"build-translation-write-metadata"
|
createTranslations,
|
||||||
|
writeTranslationMetaData
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
gulp.task(
|
gulp.task(
|
||||||
"build-supervisor-translations",
|
"build-supervisor-translations",
|
||||||
gulp.series(
|
gulp.series(toggleSupervisorFragment, "build-translations")
|
||||||
gulp.parallel(
|
|
||||||
"fetch-nightly-translations",
|
|
||||||
gulp.series("clean-translations", "ensure-translations-build-dir")
|
|
||||||
),
|
|
||||||
gulp.parallel("create-test-metadata", "create-test-translation"),
|
|
||||||
"build-master-translation",
|
|
||||||
"build-merged-translations",
|
|
||||||
"build-translation-fragment-supervisor",
|
|
||||||
"build-translation-flatten-supervisor",
|
|
||||||
"build-translation-fingerprints",
|
|
||||||
"build-translation-write-metadata"
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
@@ -99,7 +99,7 @@ gulp.task("webpack-watch-app", () => {
|
|||||||
).watch({ poll: isWsl }, doneHandler());
|
).watch({ poll: isWsl }, doneHandler());
|
||||||
gulp.watch(
|
gulp.watch(
|
||||||
path.join(paths.translations_src, "en.json"),
|
path.join(paths.translations_src, "en.json"),
|
||||||
gulp.series("create-translations", "copy-translations-app")
|
gulp.series("build-translations", "copy-translations-app")
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -1,16 +0,0 @@
|
|||||||
const path = require("path");
|
|
||||||
const fs = require("fs");
|
|
||||||
|
|
||||||
// Helper function to map recursively over files in a folder and it's subfolders
|
|
||||||
module.exports.mapFiles = function mapFiles(startPath, filter, mapFunc) {
|
|
||||||
const files = fs.readdirSync(startPath);
|
|
||||||
for (let i = 0; i < files.length; i++) {
|
|
||||||
const filename = path.join(startPath, files[i]);
|
|
||||||
const stat = fs.lstatSync(filename);
|
|
||||||
if (stat.isDirectory()) {
|
|
||||||
mapFiles(filename, filter, mapFunc);
|
|
||||||
} else if (filename.indexOf(filter) >= 0) {
|
|
||||||
mapFunc(filename);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
@@ -10,6 +10,7 @@ const WebpackBar = require("webpackbar");
|
|||||||
const {
|
const {
|
||||||
TransformAsyncModulesPlugin,
|
TransformAsyncModulesPlugin,
|
||||||
} = require("transform-async-modules-webpack-plugin");
|
} = require("transform-async-modules-webpack-plugin");
|
||||||
|
const { dependencies } = require("../package.json");
|
||||||
const paths = require("./paths.cjs");
|
const paths = require("./paths.cjs");
|
||||||
const bundle = require("./bundle.cjs");
|
const bundle = require("./bundle.cjs");
|
||||||
|
|
||||||
@@ -156,11 +157,15 @@ const createWebpackConfig = ({
|
|||||||
transform: (stats) => JSON.stringify(filterStats(stats)),
|
transform: (stats) => JSON.stringify(filterStats(stats)),
|
||||||
}),
|
}),
|
||||||
!latestBuild &&
|
!latestBuild &&
|
||||||
new TransformAsyncModulesPlugin({ browserslistEnv: "legacy" }),
|
new TransformAsyncModulesPlugin({
|
||||||
|
browserslistEnv: "legacy",
|
||||||
|
runtime: { version: dependencies["@babel/runtime"] },
|
||||||
|
}),
|
||||||
].filter(Boolean),
|
].filter(Boolean),
|
||||||
resolve: {
|
resolve: {
|
||||||
extensions: [".ts", ".js", ".json"],
|
extensions: [".ts", ".js", ".json"],
|
||||||
alias: {
|
alias: {
|
||||||
|
"lit/static-html$": "lit/static-html.js",
|
||||||
"lit/decorators$": "lit/decorators.js",
|
"lit/decorators$": "lit/decorators.js",
|
||||||
"lit/directive$": "lit/directive.js",
|
"lit/directive$": "lit/directive.js",
|
||||||
"lit/directives/until$": "lit/directives/until.js",
|
"lit/directives/until$": "lit/directives/until.js",
|
||||||
|
@@ -28,7 +28,7 @@ class HcLaunchScreen extends LitElement {
|
|||||||
:host {
|
:host {
|
||||||
display: block;
|
display: block;
|
||||||
height: 100vh;
|
height: 100vh;
|
||||||
background-color: white;
|
background-color: #f2f4f9;
|
||||||
font-size: 24px;
|
font-size: 24px;
|
||||||
}
|
}
|
||||||
.container {
|
.container {
|
||||||
@@ -43,6 +43,9 @@ class HcLaunchScreen extends LitElement {
|
|||||||
max-width: 80%;
|
max-width: 80%;
|
||||||
object-fit: cover;
|
object-fit: cover;
|
||||||
}
|
}
|
||||||
|
.status {
|
||||||
|
color: #1d2126;
|
||||||
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -270,7 +270,7 @@ export class HcMain extends HassElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this._error = undefined;
|
this._error = undefined;
|
||||||
if (msg.urlPath === "lovelace") {
|
if (msg.urlPath === "lovelace" || msg.urlPath === undefined) {
|
||||||
msg.urlPath = null;
|
msg.urlPath = null;
|
||||||
}
|
}
|
||||||
this._lovelacePath = msg.viewPath;
|
this._lovelacePath = msg.viewPath;
|
||||||
|
@@ -4,6 +4,7 @@ import { energyEntities } from "../stubs/entities";
|
|||||||
import { DemoConfig } from "./types";
|
import { DemoConfig } from "./types";
|
||||||
|
|
||||||
export const demoConfigs: Array<() => Promise<DemoConfig>> = [
|
export const demoConfigs: Array<() => Promise<DemoConfig>> = [
|
||||||
|
() => import("./sections").then((mod) => mod.demoSections),
|
||||||
() => import("./arsaboo").then((mod) => mod.demoArsaboo),
|
() => import("./arsaboo").then((mod) => mod.demoArsaboo),
|
||||||
() => import("./teachingbirds").then((mod) => mod.demoTeachingbirds),
|
() => import("./teachingbirds").then((mod) => mod.demoTeachingbirds),
|
||||||
() => import("./kernehed").then((mod) => mod.demoKernehed),
|
() => import("./kernehed").then((mod) => mod.demoKernehed),
|
||||||
|
16
demo/src/configs/sections/description.ts
Normal file
16
demo/src/configs/sections/description.ts
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
import { html } from "lit";
|
||||||
|
import { DemoConfig } from "../types";
|
||||||
|
|
||||||
|
export const demoLovelaceDescription: DemoConfig["description"] = (
|
||||||
|
localize
|
||||||
|
) => html`
|
||||||
|
<p>
|
||||||
|
${localize("ui.panel.page-demo.config.sections.description", {
|
||||||
|
blog_post: html`<a
|
||||||
|
href="https://www.home-assistant.io/blog/2024/03/04/dashboard-chapter-1/"
|
||||||
|
target="_blank"
|
||||||
|
>${localize("ui.panel.page-demo.config.sections.description_blog_post")}
|
||||||
|
</a>`,
|
||||||
|
})}
|
||||||
|
</p>
|
||||||
|
`;
|
474
demo/src/configs/sections/entities.ts
Normal file
474
demo/src/configs/sections/entities.ts
Normal file
@@ -0,0 +1,474 @@
|
|||||||
|
import { convertEntities } from "../../../../src/fake_data/entity";
|
||||||
|
import { DemoConfig } from "../types";
|
||||||
|
|
||||||
|
export const demoEntitiesSections: DemoConfig["entities"] = () =>
|
||||||
|
convertEntities({
|
||||||
|
"cover.living_room_garden_shutter": {
|
||||||
|
entity_id: "cover.living_room_garden_shutter",
|
||||||
|
state: "open",
|
||||||
|
attributes: {
|
||||||
|
current_position: 100,
|
||||||
|
device_class: "shutter",
|
||||||
|
friendly_name: "Living room garden shutter",
|
||||||
|
supported_features: 15,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"cover.living_room_graveyard_shutter": {
|
||||||
|
entity_id: "cover.living_room_graveyard_shutter",
|
||||||
|
state: "open",
|
||||||
|
attributes: {
|
||||||
|
current_position: 100,
|
||||||
|
device_class: "shutter",
|
||||||
|
friendly_name: "Living room graveyard shutter",
|
||||||
|
supported_features: 15,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"cover.living_room_left_shutter": {
|
||||||
|
entity_id: "cover.living_room_left_shutter",
|
||||||
|
state: "open",
|
||||||
|
attributes: {
|
||||||
|
current_position: 100,
|
||||||
|
device_class: "shutter",
|
||||||
|
friendly_name: "Living room left shutter",
|
||||||
|
supported_features: 15,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"cover.living_room_right_shutter": {
|
||||||
|
entity_id: "cover.living_room_right_shutter",
|
||||||
|
state: "open",
|
||||||
|
attributes: {
|
||||||
|
current_position: 100,
|
||||||
|
device_class: "shutter",
|
||||||
|
friendly_name: "Living room right shutter",
|
||||||
|
supported_features: 15,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.floor_lamp": {
|
||||||
|
entity_id: "light.floor_lamp",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
min_color_temp_kelvin: 2000,
|
||||||
|
max_color_temp_kelvin: 6535,
|
||||||
|
min_mireds: 153,
|
||||||
|
max_mireds: 500,
|
||||||
|
supported_color_modes: ["color_temp", "xy"],
|
||||||
|
color_mode: "color_temp",
|
||||||
|
brightness: 178,
|
||||||
|
color_temp_kelvin: 2583,
|
||||||
|
color_temp: 387,
|
||||||
|
hs_color: [28.664, 69.597],
|
||||||
|
rgb_color: [255, 162, 77],
|
||||||
|
xy_color: [0.538, 0.389],
|
||||||
|
icon: "mdi:floor-lamp",
|
||||||
|
friendly_name: "Floor lamp",
|
||||||
|
supported_features: 44,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.living_room_spotlights": {
|
||||||
|
entity_id: "light.living_room_spotlights",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
|
color_mode: "brightness",
|
||||||
|
brightness: 126,
|
||||||
|
icon: "mdi:ceiling-light-multiple",
|
||||||
|
friendly_name: "Living room spotlights",
|
||||||
|
supported_features: 32,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.bar_lamp": {
|
||||||
|
entity_id: "light.bar_lamp",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
min_color_temp_kelvin: 2202,
|
||||||
|
max_color_temp_kelvin: 4504,
|
||||||
|
min_mireds: 222,
|
||||||
|
max_mireds: 454,
|
||||||
|
effect_list: ["None", "candle"],
|
||||||
|
supported_color_modes: ["color_temp"],
|
||||||
|
effect: null,
|
||||||
|
color_mode: null,
|
||||||
|
brightness: null,
|
||||||
|
color_temp_kelvin: null,
|
||||||
|
color_temp: null,
|
||||||
|
hs_color: null,
|
||||||
|
rgb_color: null,
|
||||||
|
xy_color: null,
|
||||||
|
mode: "normal",
|
||||||
|
dynamics: "none",
|
||||||
|
icon: "mdi:lightbulb-variant",
|
||||||
|
friendly_name: "Bar lamp",
|
||||||
|
supported_features: 44,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.living_room_temperature": {
|
||||||
|
entity_id: "sensor.living_room_temperature",
|
||||||
|
state: "22.8",
|
||||||
|
attributes: {
|
||||||
|
state_class: "measurement",
|
||||||
|
unit_of_measurement: "°C",
|
||||||
|
device_class: "temperature",
|
||||||
|
friendly_name: "Living room Temperature",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"media_player.living_room_nest_mini": {
|
||||||
|
entity_id: "media_player.living_room_nest_mini",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
device_class: "speaker",
|
||||||
|
friendly_name: "Living room Nest Mini",
|
||||||
|
supported_features: 152461,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"cover.kitchen_shutter": {
|
||||||
|
entity_id: "cover.kitchen_shutter",
|
||||||
|
state: "open",
|
||||||
|
attributes: {
|
||||||
|
current_position: 100,
|
||||||
|
device_class: "shutter",
|
||||||
|
friendly_name: "Kitchen shutter ",
|
||||||
|
supported_features: 15,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.kitchen_spotlights": {
|
||||||
|
entity_id: "light.kitchen_spotlights",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
|
color_mode: null,
|
||||||
|
brightness: null,
|
||||||
|
icon: "mdi:ceiling-light-multiple",
|
||||||
|
friendly_name: "Kitchen spotlights ",
|
||||||
|
supported_features: 32,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.worktop_spotlights": {
|
||||||
|
entity_id: "light.worktop_spotlights",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
|
color_mode: null,
|
||||||
|
brightness: null,
|
||||||
|
icon: "mdi:ceiling-light-multiple",
|
||||||
|
friendly_name: "Worktop spotlights ",
|
||||||
|
supported_features: 32,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"binary_sensor.fridge_door": {
|
||||||
|
entity_id: "binary_sensor.fridge_door",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
device_class: "door",
|
||||||
|
icon: "mdi:fridge",
|
||||||
|
friendly_name: "Fridge door",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"media_player.kitchen_nest_audio": {
|
||||||
|
entity_id: "media_player.kitchen_nest_audio",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
device_class: "speaker",
|
||||||
|
friendly_name: "Kitchen Nest Audio",
|
||||||
|
supported_features: 152461,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"binary_sensor.tesla_wall_connector_vehicle_connected": {
|
||||||
|
entity_id: "binary_sensor.tesla_wall_connector_vehicle_connected",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
device_class: "plug",
|
||||||
|
friendly_name: "Wall Connector Vehicle connected",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.tesla_wall_connector_session_energy": {
|
||||||
|
entity_id: "sensor.tesla_wall_connector_session_energy",
|
||||||
|
state: "16.3",
|
||||||
|
attributes: {
|
||||||
|
state_class: "total_increasing",
|
||||||
|
unit_of_measurement: "kWh",
|
||||||
|
device_class: "energy",
|
||||||
|
friendly_name: "Tesla Wall Connector Session energy",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.electric_meter_power": {
|
||||||
|
entity_id: "sensor.electric_meter_power",
|
||||||
|
state: "797.86",
|
||||||
|
attributes: {
|
||||||
|
state_class: "measurement",
|
||||||
|
unit_of_measurement: "W",
|
||||||
|
device_class: "power",
|
||||||
|
icon: "mdi:meter-electric",
|
||||||
|
friendly_name: "Electric meter Power",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.eletric_meter_voltage": {
|
||||||
|
entity_id: "sensor.eletric_meter_voltage",
|
||||||
|
state: "232.19",
|
||||||
|
attributes: {
|
||||||
|
state_class: "measurement",
|
||||||
|
unit_of_measurement: "V",
|
||||||
|
device_class: "voltage",
|
||||||
|
friendly_name: "Electric meter voltage",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.electricity_maps_grid_fossil_fuel_percentage": {
|
||||||
|
entity_id: "sensor.electricity_maps_grid_fossil_fuel_percentage",
|
||||||
|
state: "9.84",
|
||||||
|
attributes: {
|
||||||
|
state_class: "measurement",
|
||||||
|
country_code: "FR",
|
||||||
|
unit_of_measurement: "%",
|
||||||
|
attribution: "Data provided by Electricity Maps",
|
||||||
|
icon: "mdi:barrel",
|
||||||
|
friendly_name: "Electricity Maps Grid fossil fuel percentage",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.electricity_maps_co2_intensity": {
|
||||||
|
entity_id: "sensor.electricity_maps_co2_intensity",
|
||||||
|
state: "62.0",
|
||||||
|
attributes: {
|
||||||
|
state_class: "measurement",
|
||||||
|
country_code: "FR",
|
||||||
|
unit_of_measurement: "gCO2eq/kWh",
|
||||||
|
attribution: "Data provided by Electricity Maps",
|
||||||
|
friendly_name: "Electricity Maps CO2 intensity",
|
||||||
|
icon: "mdi:molecule-co2",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sun.sun": {
|
||||||
|
entity_id: "sun.sun",
|
||||||
|
state: "above_horizon",
|
||||||
|
attributes: {
|
||||||
|
next_dawn: "2024-03-05T05:50:21.964405+00:00",
|
||||||
|
next_dusk: "2024-03-04T18:08:54.311334+00:00",
|
||||||
|
next_midnight: "2024-03-05T00:00:00+00:00",
|
||||||
|
next_noon: "2024-03-05T12:00:05+00:00",
|
||||||
|
next_rising: "2024-03-05T06:23:42.739159+00:00",
|
||||||
|
next_setting: "2024-03-04T17:35:26.271171+00:00",
|
||||||
|
elevation: 30.38,
|
||||||
|
azimuth: 204.42,
|
||||||
|
rising: false,
|
||||||
|
friendly_name: "Sun",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.rain": {
|
||||||
|
entity_id: "sensor.moon_phase",
|
||||||
|
state: "7.2",
|
||||||
|
attributes: {
|
||||||
|
state_class: "total_increasing",
|
||||||
|
unit_of_measurement: "mm",
|
||||||
|
device_class: "precipitation",
|
||||||
|
friendly_name: "Rain",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"climate.ground_floor": {
|
||||||
|
entity_id: "climate.ground_floor",
|
||||||
|
state: "heat",
|
||||||
|
attributes: {
|
||||||
|
hvac_modes: ["auto", "heat", "off"],
|
||||||
|
min_temp: 7,
|
||||||
|
max_temp: 35,
|
||||||
|
preset_modes: [
|
||||||
|
"comfort",
|
||||||
|
"away",
|
||||||
|
"eco",
|
||||||
|
"frost_protection",
|
||||||
|
"external",
|
||||||
|
"home",
|
||||||
|
],
|
||||||
|
current_temperature: 20.8,
|
||||||
|
temperature: 21,
|
||||||
|
preset_mode: "comfort",
|
||||||
|
icon: "mdi:home-floor-0",
|
||||||
|
friendly_name: "Ground floor Thermostat",
|
||||||
|
supported_features: 401,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"climate.first_floor": {
|
||||||
|
entity_id: "climate.first_floor",
|
||||||
|
state: "heat",
|
||||||
|
attributes: {
|
||||||
|
hvac_modes: ["auto", "heat", "off"],
|
||||||
|
min_temp: 7,
|
||||||
|
max_temp: 35,
|
||||||
|
preset_modes: [
|
||||||
|
"comfort",
|
||||||
|
"away",
|
||||||
|
"eco",
|
||||||
|
"frost_protection",
|
||||||
|
"external",
|
||||||
|
"home",
|
||||||
|
],
|
||||||
|
current_temperature: 21.7,
|
||||||
|
temperature: 21,
|
||||||
|
preset_mode: "comfort",
|
||||||
|
icon: "mdi:home-floor-1",
|
||||||
|
friendly_name: "First floor Thermostat",
|
||||||
|
supported_features: 401,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"cover.study_shutter": {
|
||||||
|
entity_id: "cover.study_shutter",
|
||||||
|
state: "open",
|
||||||
|
attributes: {
|
||||||
|
current_position: 100,
|
||||||
|
device_class: "shutter",
|
||||||
|
friendly_name: "Study shutter",
|
||||||
|
supported_features: 15,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.study_spotlights": {
|
||||||
|
entity_id: "light.study_spotlights",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
|
color_mode: null,
|
||||||
|
brightness: null,
|
||||||
|
icon: "mdi:ceiling-light-multiple",
|
||||||
|
friendly_name: "Study spotlights",
|
||||||
|
supported_features: 32,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"media_player.study_nest_hub": {
|
||||||
|
entity_id: "media_player.study_nest_hub",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
friendly_name: "Study Nest Hub",
|
||||||
|
supported_features: 152461,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.standing_desk_height": {
|
||||||
|
entity_id: "sensor.standing_desk_height",
|
||||||
|
state: "72",
|
||||||
|
attributes: {
|
||||||
|
unit_of_measurement: "cm",
|
||||||
|
icon: "mdi:tape-measure",
|
||||||
|
friendly_name: "Standing desk Height",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.outdoor_light": {
|
||||||
|
entity_id: "light.outdoor_light",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
|
color_mode: null,
|
||||||
|
brightness: 255,
|
||||||
|
icon: "mdi:outdoor-lamp",
|
||||||
|
friendly_name: "Outdoor light",
|
||||||
|
supported_features: 32,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.flood_light": {
|
||||||
|
entity_id: "light.flood_light",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
effect_list: ["None", "candle"],
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
|
effect: null,
|
||||||
|
color_mode: null,
|
||||||
|
brightness: null,
|
||||||
|
mode: "normal",
|
||||||
|
dynamics: "none",
|
||||||
|
icon: "mdi:light-flood-down",
|
||||||
|
friendly_name: "Flood light",
|
||||||
|
supported_features: 44,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.outdoor_motion_sensor_temperature": {
|
||||||
|
entity_id: "sensor.outdoor_motion_sensor_temperature",
|
||||||
|
state: "10.2",
|
||||||
|
attributes: {
|
||||||
|
state_class: "measurement",
|
||||||
|
unit_of_measurement: "°C",
|
||||||
|
device_class: "temperature",
|
||||||
|
friendly_name: "Outdoor motion sensor Temperature",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"binary_sensor.outdoor_motion_sensor_motion": {
|
||||||
|
entity_id: "binary_sensor.outdoor_motion_sensor_motion",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
device_class: "motion",
|
||||||
|
friendly_name: "Outdoor motion sensor Motion",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"sensor.outdoor_motion_sensor_illuminance": {
|
||||||
|
entity_id: "sensor.outdoor_motion_sensor_illuminance",
|
||||||
|
state: "555",
|
||||||
|
attributes: {
|
||||||
|
state_class: "measurement",
|
||||||
|
light_level: 27444,
|
||||||
|
unit_of_measurement: "lx",
|
||||||
|
device_class: "illuminance",
|
||||||
|
friendly_name: "Outdoor motion sensor Illuminance",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"automation.home_assistant_auto_update": {
|
||||||
|
entity_id: "automation.home_assistant_auto_update",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
id: "1700669321947",
|
||||||
|
last_triggered: "2024-02-29T18:02:05.343139+00:00",
|
||||||
|
mode: "queued",
|
||||||
|
current: 0,
|
||||||
|
max: 50,
|
||||||
|
icon: "mdi:auto-mode",
|
||||||
|
friendly_name: "Home Assistant Auto-update",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"update.home_assistant_operating_system_update": {
|
||||||
|
entity_id: "update.home_assistant_operating_system_update",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
auto_update: false,
|
||||||
|
installed_version: "12.1",
|
||||||
|
in_progress: false,
|
||||||
|
latest_version: "12.1",
|
||||||
|
release_summary: null,
|
||||||
|
release_url:
|
||||||
|
"https://github.com/home-assistant/operating-system/commits/dev",
|
||||||
|
skipped_version: null,
|
||||||
|
title: "Home Assistant Operating System",
|
||||||
|
entity_picture:
|
||||||
|
"https://brands.home-assistant.io/homeassistant/icon.png",
|
||||||
|
friendly_name: "Home Assistant Operating System Update",
|
||||||
|
supported_features: 3,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"update.home_assistant_supervisor_update": {
|
||||||
|
entity_id: "update.home_assistant_supervisor_update",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
auto_update: true,
|
||||||
|
installed_version: "2024.02.2",
|
||||||
|
in_progress: false,
|
||||||
|
latest_version: "2024.02.2",
|
||||||
|
release_summary: null,
|
||||||
|
release_url:
|
||||||
|
"https://github.com/home-assistant/supervisor/commits/main",
|
||||||
|
skipped_version: null,
|
||||||
|
title: "Home Assistant Supervisor",
|
||||||
|
entity_picture: "https://brands.home-assistant.io/hassio/icon.png",
|
||||||
|
friendly_name: "Home Assistant Supervisor Update",
|
||||||
|
supported_features: 1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"update.home_assistant_core_update": {
|
||||||
|
entity_id: "update.home_assistant_supervisor_update",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
auto_update: false,
|
||||||
|
installed_version: "2024.4.0",
|
||||||
|
in_progress: false,
|
||||||
|
latest_version: "2024.4.0",
|
||||||
|
release_summary: null,
|
||||||
|
release_url: "https://github.com/home-assistant/core/commits/dev",
|
||||||
|
skipped_version: null,
|
||||||
|
title: "Home Assistant Core",
|
||||||
|
entity_picture:
|
||||||
|
"https://brands.home-assistant.io/homeassistant/icon.png",
|
||||||
|
friendly_name: "Home Assistant Core Update",
|
||||||
|
supported_features: 11,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
14
demo/src/configs/sections/index.ts
Normal file
14
demo/src/configs/sections/index.ts
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import { DemoConfig } from "../types";
|
||||||
|
import { demoLovelaceDescription } from "./description";
|
||||||
|
import { demoEntitiesSections } from "./entities";
|
||||||
|
import { demoLovelaceSections } from "./lovelace";
|
||||||
|
|
||||||
|
export const demoSections: DemoConfig = {
|
||||||
|
authorName: "Home Assistant",
|
||||||
|
authorUrl: "https://github.com/home-assistant/frontend/",
|
||||||
|
name: "Home Demo",
|
||||||
|
description: demoLovelaceDescription,
|
||||||
|
lovelace: demoLovelaceSections,
|
||||||
|
entities: demoEntitiesSections,
|
||||||
|
theme: () => ({}),
|
||||||
|
};
|
281
demo/src/configs/sections/lovelace.ts
Normal file
281
demo/src/configs/sections/lovelace.ts
Normal file
@@ -0,0 +1,281 @@
|
|||||||
|
import { DemoConfig } from "../types";
|
||||||
|
|
||||||
|
export const demoLovelaceSections: DemoConfig["lovelace"] = () => ({
|
||||||
|
title: "Home Assistant Demo",
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
type: "sections",
|
||||||
|
title: "Demo",
|
||||||
|
path: "home",
|
||||||
|
icon: "mdi:home-assistant",
|
||||||
|
sections: [
|
||||||
|
{
|
||||||
|
title: "Welcome 👋",
|
||||||
|
cards: [{ type: "custom:ha-demo-card" }],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
cards: [
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "cover.living_room_garden_shutter",
|
||||||
|
name: "Garden",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "cover.living_room_graveyard_shutter",
|
||||||
|
name: "Rear",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "cover.living_room_left_shutter",
|
||||||
|
name: "Left",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "cover.living_room_right_shutter",
|
||||||
|
name: "Right",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "light.floor_lamp",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "light.living_room_spotlights",
|
||||||
|
name: "Spotlights",
|
||||||
|
features: [
|
||||||
|
{
|
||||||
|
type: "light-brightness",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "light.bar_lamp",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
graph: "line",
|
||||||
|
type: "sensor",
|
||||||
|
entity: "sensor.living_room_temperature",
|
||||||
|
detail: 1,
|
||||||
|
name: "Temperature",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "media_player.living_room_nest_mini",
|
||||||
|
name: "Nest Mini",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
title: "🛋️ Living room ",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "grid",
|
||||||
|
cards: [
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "cover.kitchen_shutter",
|
||||||
|
name: "Shutter",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "light.kitchen_spotlights",
|
||||||
|
name: "Spotlights",
|
||||||
|
features: [
|
||||||
|
{
|
||||||
|
type: "light-brightness",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "light.worktop_spotlights",
|
||||||
|
name: "Worktop",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "binary_sensor.fridge_door",
|
||||||
|
name: "Fridge",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "media_player.kitchen_nest_audio",
|
||||||
|
name: "Nest Audio",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
title: "👩🍳 Kitchen",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "grid",
|
||||||
|
cards: [
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "binary_sensor.tesla_wall_connector_vehicle_connected",
|
||||||
|
name: "EV",
|
||||||
|
icon: "mdi:car",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sensor.tesla_wall_connector_session_energy",
|
||||||
|
name: "Last charge",
|
||||||
|
color: "green",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sensor.electric_meter_power",
|
||||||
|
color: "deep-orange",
|
||||||
|
name: "Home power",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sensor.eletric_meter_voltage",
|
||||||
|
name: "Voltage",
|
||||||
|
color: "deep-orange",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sensor.electricity_maps_grid_fossil_fuel_percentage",
|
||||||
|
name: "Fossil fuel",
|
||||||
|
color: "brown",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sensor.electricity_maps_co2_intensity",
|
||||||
|
name: "CO2 Intensity",
|
||||||
|
color: "dark-grey",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
title: "⚡️ Energy",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "grid",
|
||||||
|
cards: [
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sun.sun",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sensor.rain",
|
||||||
|
color: "blue",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
features: [
|
||||||
|
{
|
||||||
|
type: "target-temperature",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
type: "tile",
|
||||||
|
name: "Downstairs",
|
||||||
|
entity: "climate.ground_floor",
|
||||||
|
state_content: ["preset_mode", "current_temperature"],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
features: [
|
||||||
|
{
|
||||||
|
type: "target-temperature",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
type: "tile",
|
||||||
|
name: "Upstairs",
|
||||||
|
entity: "climate.first_floor",
|
||||||
|
state_content: ["preset_mode", "current_temperature"],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
title: "🌤️ Climate",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "grid",
|
||||||
|
cards: [
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "cover.study_shutter",
|
||||||
|
name: "Shutter",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "light.study_spotlights",
|
||||||
|
name: "Spotlights",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "media_player.study_nest_hub",
|
||||||
|
name: "Nest Hub",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sensor.standing_desk_height",
|
||||||
|
name: "Desk",
|
||||||
|
color: "brown",
|
||||||
|
icon: "mdi:desk",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
title: "🧑💻 Study",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "grid",
|
||||||
|
cards: [
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "light.outdoor_light",
|
||||||
|
name: "Door light",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "light.flood_light",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
graph: "line",
|
||||||
|
type: "sensor",
|
||||||
|
entity: "sensor.outdoor_motion_sensor_temperature",
|
||||||
|
detail: 1,
|
||||||
|
name: "Temperature",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "binary_sensor.outdoor_motion_sensor_motion",
|
||||||
|
name: "Motion",
|
||||||
|
color: "blue",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "sensor.outdoor_motion_sensor_illuminance",
|
||||||
|
color: "amber",
|
||||||
|
name: "Illuminance",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
title: "🌳 Outdoor",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "grid",
|
||||||
|
cards: [
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "automation.home_assistant_auto_update",
|
||||||
|
name: "Auto-update",
|
||||||
|
color: "green",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "update.home_assistant_operating_system_update",
|
||||||
|
name: "OS",
|
||||||
|
icon: "mdi:home-assistant",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "update.home_assistant_supervisor_update",
|
||||||
|
icon: "mdi:home-assistant",
|
||||||
|
name: "Supervisor",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "tile",
|
||||||
|
entity: "update.home_assistant_core_update",
|
||||||
|
name: "Core",
|
||||||
|
icon: "mdi:home-assistant",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
title: "🎉 Updates",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
@@ -1,3 +1,4 @@
|
|||||||
|
import { TemplateResult } from "lit";
|
||||||
import { LocalizeFunc } from "../../../src/common/translations/localize";
|
import { LocalizeFunc } from "../../../src/common/translations/localize";
|
||||||
import { LovelaceConfig } from "../../../src/data/lovelace/config/types";
|
import { LovelaceConfig } from "../../../src/data/lovelace/config/types";
|
||||||
import { Entity } from "../../../src/fake_data/entity";
|
import { Entity } from "../../../src/fake_data/entity";
|
||||||
@@ -7,6 +8,9 @@ export interface DemoConfig {
|
|||||||
name: string;
|
name: string;
|
||||||
authorName: string;
|
authorName: string;
|
||||||
authorUrl: string;
|
authorUrl: string;
|
||||||
|
description?:
|
||||||
|
| string
|
||||||
|
| ((localize: LocalizeFunc) => string | TemplateResult<1>);
|
||||||
lovelace: (localize: LocalizeFunc) => LovelaceConfig;
|
lovelace: (localize: LocalizeFunc) => LovelaceConfig;
|
||||||
entities: (localize: LocalizeFunc) => Entity[];
|
entities: (localize: LocalizeFunc) => Entity[];
|
||||||
theme: () => Record<string, string> | null;
|
theme: () => Record<string, string> | null;
|
||||||
|
@@ -39,32 +39,51 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
<div class="picker">
|
<div class="picker">
|
||||||
<div class="label">
|
<div class="label">
|
||||||
${this._switching
|
${this._switching
|
||||||
? html`<ha-circular-progress
|
? html`
|
||||||
indeterminate
|
<ha-circular-progress indeterminate></ha-circular-progress>
|
||||||
></ha-circular-progress>`
|
`
|
||||||
: until(
|
: until(
|
||||||
selectedDemoConfig.then(
|
selectedDemoConfig.then(
|
||||||
(conf) => html`
|
(conf) => html`
|
||||||
${conf.name}
|
${conf.name}
|
||||||
<small>
|
<small>
|
||||||
<a target="_blank" href=${conf.authorUrl}>
|
${this.hass.localize(
|
||||||
${this.hass.localize(
|
"ui.panel.page-demo.cards.demo.demo_by",
|
||||||
"ui.panel.page-demo.cards.demo.demo_by",
|
{
|
||||||
{ name: conf.authorName }
|
name: html`
|
||||||
)}
|
<a target="_blank" href=${conf.authorUrl}>
|
||||||
</a>
|
${conf.authorName}
|
||||||
|
</a>
|
||||||
|
`,
|
||||||
|
}
|
||||||
|
)}
|
||||||
</small>
|
</small>
|
||||||
`
|
`
|
||||||
),
|
),
|
||||||
""
|
""
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<mwc-button @click=${this._nextConfig} .disabled=${this._switching}>
|
<mwc-button @click=${this._nextConfig} .disabled=${this._switching}>
|
||||||
${this.hass.localize("ui.panel.page-demo.cards.demo.next_demo")}
|
${this.hass.localize("ui.panel.page-demo.cards.demo.next_demo")}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
</div>
|
</div>
|
||||||
<div class="content small-hidden">
|
<div class="content">
|
||||||
${this.hass.localize("ui.panel.page-demo.cards.demo.introduction")}
|
<p class="small-hidden">
|
||||||
|
${this.hass.localize("ui.panel.page-demo.cards.demo.introduction")}
|
||||||
|
</p>
|
||||||
|
${until(
|
||||||
|
selectedDemoConfig.then((conf) => {
|
||||||
|
if (typeof conf.description === "function") {
|
||||||
|
return conf.description(this.hass.localize);
|
||||||
|
}
|
||||||
|
if (conf.description) {
|
||||||
|
return html`<p>${conf.description}</p>`;
|
||||||
|
}
|
||||||
|
return nothing;
|
||||||
|
}),
|
||||||
|
nothing
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div class="actions small-hidden">
|
<div class="actions small-hidden">
|
||||||
<a href="https://www.home-assistant.io" target="_blank">
|
<a href="https://www.home-assistant.io" target="_blank">
|
||||||
@@ -108,6 +127,7 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
css`
|
css`
|
||||||
a {
|
a {
|
||||||
color: var(--primary-color);
|
color: var(--primary-color);
|
||||||
|
display: inline-block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.actions a {
|
.actions a {
|
||||||
@@ -115,7 +135,11 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.content {
|
.content {
|
||||||
padding: 16px;
|
padding: 0 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content p {
|
||||||
|
margin: 16px 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.picker {
|
.picker {
|
||||||
@@ -138,9 +162,8 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.actions {
|
.actions {
|
||||||
padding-left: 8px;
|
padding: 0px 8px 4px 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media only screen and (max-width: 500px) {
|
@media only screen and (max-width: 500px) {
|
||||||
.small-hidden {
|
.small-hidden {
|
||||||
display: none;
|
display: none;
|
||||||
|
@@ -10,6 +10,7 @@ import {
|
|||||||
import { HomeAssistantAppEl } from "../../src/layouts/home-assistant";
|
import { HomeAssistantAppEl } from "../../src/layouts/home-assistant";
|
||||||
import { HomeAssistant } from "../../src/types";
|
import { HomeAssistant } from "../../src/types";
|
||||||
import { selectedDemoConfig } from "./configs/demo-configs";
|
import { selectedDemoConfig } from "./configs/demo-configs";
|
||||||
|
import { mockAreaRegistry } from "./stubs/area_registry";
|
||||||
import { mockAuth } from "./stubs/auth";
|
import { mockAuth } from "./stubs/auth";
|
||||||
import { mockConfigEntries } from "./stubs/config_entries";
|
import { mockConfigEntries } from "./stubs/config_entries";
|
||||||
import { mockEnergy } from "./stubs/energy";
|
import { mockEnergy } from "./stubs/energy";
|
||||||
@@ -23,10 +24,10 @@ import { mockLovelace } from "./stubs/lovelace";
|
|||||||
import { mockMediaPlayer } from "./stubs/media_player";
|
import { mockMediaPlayer } from "./stubs/media_player";
|
||||||
import { mockPersistentNotification } from "./stubs/persistent_notification";
|
import { mockPersistentNotification } from "./stubs/persistent_notification";
|
||||||
import { mockRecorder } from "./stubs/recorder";
|
import { mockRecorder } from "./stubs/recorder";
|
||||||
import { mockTodo } from "./stubs/todo";
|
|
||||||
import { mockSensor } from "./stubs/sensor";
|
import { mockSensor } from "./stubs/sensor";
|
||||||
import { mockSystemLog } from "./stubs/system_log";
|
import { mockSystemLog } from "./stubs/system_log";
|
||||||
import { mockTemplate } from "./stubs/template";
|
import { mockTemplate } from "./stubs/template";
|
||||||
|
import { mockTodo } from "./stubs/todo";
|
||||||
import { mockTranslations } from "./stubs/translations";
|
import { mockTranslations } from "./stubs/translations";
|
||||||
|
|
||||||
@customElement("ha-demo")
|
@customElement("ha-demo")
|
||||||
@@ -62,6 +63,7 @@ export class HaDemo extends HomeAssistantAppEl {
|
|||||||
mockEnergy(hass);
|
mockEnergy(hass);
|
||||||
mockPersistentNotification(hass);
|
mockPersistentNotification(hass);
|
||||||
mockConfigEntries(hass);
|
mockConfigEntries(hass);
|
||||||
|
mockAreaRegistry(hass);
|
||||||
mockEntityRegistry(hass, [
|
mockEntityRegistry(hass, [
|
||||||
{
|
{
|
||||||
config_entry_id: "co2signal",
|
config_entry_id: "co2signal",
|
||||||
@@ -72,6 +74,8 @@ export class HaDemo extends HomeAssistantAppEl {
|
|||||||
id: "sensor.co2_intensity",
|
id: "sensor.co2_intensity",
|
||||||
name: null,
|
name: null,
|
||||||
icon: null,
|
icon: null,
|
||||||
|
labels: [],
|
||||||
|
categories: {},
|
||||||
platform: "co2signal",
|
platform: "co2signal",
|
||||||
hidden_by: null,
|
hidden_by: null,
|
||||||
entity_category: null,
|
entity_category: null,
|
||||||
@@ -88,6 +92,8 @@ export class HaDemo extends HomeAssistantAppEl {
|
|||||||
id: "sensor.co2_intensity",
|
id: "sensor.co2_intensity",
|
||||||
name: null,
|
name: null,
|
||||||
icon: null,
|
icon: null,
|
||||||
|
labels: [],
|
||||||
|
categories: {},
|
||||||
platform: "co2signal",
|
platform: "co2signal",
|
||||||
hidden_by: null,
|
hidden_by: null,
|
||||||
entity_category: null,
|
entity_category: null,
|
||||||
|
@@ -4,4 +4,11 @@ import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|||||||
export const mockAreaRegistry = (
|
export const mockAreaRegistry = (
|
||||||
hass: MockHomeAssistant,
|
hass: MockHomeAssistant,
|
||||||
data: AreaRegistryEntry[] = []
|
data: AreaRegistryEntry[] = []
|
||||||
) => hass.mockWS("config/area_registry/list", () => data);
|
) => {
|
||||||
|
hass.mockWS("config/area_registry/list", () => data);
|
||||||
|
const areas = {};
|
||||||
|
data.forEach((area) => {
|
||||||
|
areas[area.area_id] = area;
|
||||||
|
});
|
||||||
|
hass.updateHass({ areas });
|
||||||
|
};
|
||||||
|
@@ -10,6 +10,7 @@ export const mockConfigEntries = (hass: MockHomeAssistant) => {
|
|||||||
supports_options: false,
|
supports_options: false,
|
||||||
supports_remove_device: false,
|
supports_remove_device: false,
|
||||||
supports_unload: true,
|
supports_unload: true,
|
||||||
|
supports_reconfigure: true,
|
||||||
pref_disable_new_entities: false,
|
pref_disable_new_entities: false,
|
||||||
pref_disable_polling: false,
|
pref_disable_polling: false,
|
||||||
disabled_by: null,
|
disabled_by: null,
|
||||||
|
@@ -4,4 +4,11 @@ import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|||||||
export const mockDeviceRegistry = (
|
export const mockDeviceRegistry = (
|
||||||
hass: MockHomeAssistant,
|
hass: MockHomeAssistant,
|
||||||
data: DeviceRegistryEntry[] = []
|
data: DeviceRegistryEntry[] = []
|
||||||
) => hass.mockWS("config/device_registry/list", () => data);
|
) => {
|
||||||
|
hass.mockWS("config/device_registry/list", () => data);
|
||||||
|
const devices = {};
|
||||||
|
data.forEach((device) => {
|
||||||
|
devices[device.id] = device;
|
||||||
|
});
|
||||||
|
hass.updateHass({ devices });
|
||||||
|
};
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { format, startOfToday, startOfTomorrow } from "date-fns/esm";
|
import { format, startOfToday, startOfTomorrow } from "date-fns";
|
||||||
import {
|
import {
|
||||||
EnergyInfo,
|
EnergyInfo,
|
||||||
EnergyPreferences,
|
EnergyPreferences,
|
||||||
|
7
demo/src/stubs/floor_registry.ts
Normal file
7
demo/src/stubs/floor_registry.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { FloorRegistryEntry } from "../../../src/data/floor_registry";
|
||||||
|
import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
||||||
|
|
||||||
|
export const mockFloorRegistry = (
|
||||||
|
hass: MockHomeAssistant,
|
||||||
|
data: FloorRegistryEntry[] = []
|
||||||
|
) => hass.mockWS("config/floor_registry/list", () => data);
|
7
demo/src/stubs/label_registry.ts
Normal file
7
demo/src/stubs/label_registry.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { LabelRegistryEntry } from "../../../src/data/label_registry";
|
||||||
|
import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
||||||
|
|
||||||
|
export const mockLabelRegistry = (
|
||||||
|
hass: MockHomeAssistant,
|
||||||
|
data: LabelRegistryEntry[] = []
|
||||||
|
) => hass.mockWS("config/label_registry/list", () => data);
|
@@ -17,6 +17,7 @@ export const basicTrace: DemoTrace = {
|
|||||||
{
|
{
|
||||||
path: "trigger/0",
|
path: "trigger/0",
|
||||||
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
||||||
|
changed_variables: {},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
"condition/0": [
|
"condition/0": [
|
||||||
|
@@ -17,6 +17,7 @@ export const motionLightTrace: DemoTrace = {
|
|||||||
{
|
{
|
||||||
path: "trigger/0",
|
path: "trigger/0",
|
||||||
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
||||||
|
changed_variables: {},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
"action/0": [
|
"action/0": [
|
||||||
|
@@ -136,7 +136,7 @@ export class DemoAutomationDescribeAction extends LitElement {
|
|||||||
<div class="action">
|
<div class="action">
|
||||||
<span>
|
<span>
|
||||||
${this._action
|
${this._action
|
||||||
? describeAction(this.hass, [], this._action)
|
? describeAction(this.hass, [], [], [], this._action)
|
||||||
: "<invalid YAML>"}
|
: "<invalid YAML>"}
|
||||||
</span>
|
</span>
|
||||||
<ha-yaml-editor
|
<ha-yaml-editor
|
||||||
@@ -149,7 +149,7 @@ export class DemoAutomationDescribeAction extends LitElement {
|
|||||||
${ACTIONS.map(
|
${ACTIONS.map(
|
||||||
(conf) => html`
|
(conf) => html`
|
||||||
<div class="action">
|
<div class="action">
|
||||||
<span>${describeAction(this.hass, [], conf as any)}</span>
|
<span>${describeAction(this.hass, [], [], [], conf as any)}</span>
|
||||||
<pre>${dump(conf)}</pre>
|
<pre>${dump(conf)}</pre>
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
|
@@ -21,10 +21,10 @@ const ENTITIES = [
|
|||||||
}),
|
}),
|
||||||
];
|
];
|
||||||
|
|
||||||
const conditions = [
|
const conditions: Condition[] = [
|
||||||
{ condition: "and" },
|
{ condition: "and", conditions: [] },
|
||||||
{ condition: "not" },
|
{ condition: "not", conditions: [] },
|
||||||
{ condition: "or" },
|
{ condition: "or", conditions: [] },
|
||||||
{ condition: "state", entity_id: "light.kitchen", state: "on" },
|
{ condition: "state", entity_id: "light.kitchen", state: "on" },
|
||||||
{
|
{
|
||||||
condition: "numeric_state",
|
condition: "numeric_state",
|
||||||
@@ -34,11 +34,11 @@ const conditions = [
|
|||||||
above: 20,
|
above: 20,
|
||||||
},
|
},
|
||||||
{ condition: "sun", after: "sunset" },
|
{ condition: "sun", after: "sunset" },
|
||||||
{ condition: "sun", after: "sunrise", offset: "-01:00" },
|
{ condition: "sun", after: "sunrise", before_offset: 3600 },
|
||||||
{ condition: "zone", entity_id: "device_tracker.person", zone: "zone.home" },
|
{ condition: "zone", entity_id: "device_tracker.person", zone: "zone.home" },
|
||||||
{ condition: "trigger", id: "motion" },
|
{ condition: "trigger", id: "motion" },
|
||||||
{ condition: "time" },
|
{ condition: "time" },
|
||||||
{ condition: "template" },
|
{ condition: "template", value_template: "" },
|
||||||
];
|
];
|
||||||
|
|
||||||
const initialCondition: Condition = {
|
const initialCondition: Condition = {
|
||||||
|
@@ -55,6 +55,7 @@ export class DemoAutomationTraceTimeline extends LitElement {
|
|||||||
super.firstUpdated(changedProps);
|
super.firstUpdated(changedProps);
|
||||||
const hass = provideHass(this);
|
const hass = provideHass(this);
|
||||||
hass.updateTranslations(null, "en");
|
hass.updateTranslations(null, "en");
|
||||||
|
hass.updateTranslations("config", "en");
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles() {
|
static get styles() {
|
||||||
|
@@ -60,6 +60,7 @@ export class DemoAutomationTrace extends LitElement {
|
|||||||
super.firstUpdated(changedProps);
|
super.firstUpdated(changedProps);
|
||||||
const hass = provideHass(this);
|
const hass = provideHass(this);
|
||||||
hass.updateTranslations(null, "en");
|
hass.updateTranslations(null, "en");
|
||||||
|
hass.updateTranslations("config", "en");
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles() {
|
static get styles() {
|
||||||
|
@@ -162,7 +162,7 @@ export class DemoHaBarButton extends LitElement {
|
|||||||
}
|
}
|
||||||
.custom-group {
|
.custom-group {
|
||||||
--control-button-group-thickness: 100px;
|
--control-button-group-thickness: 100px;
|
||||||
--control-button-group-border-radius: 18px;
|
--control-button-group-border-radius: 36px;
|
||||||
--control-button-group-spacing: 20px;
|
--control-button-group-spacing: 20px;
|
||||||
}
|
}
|
||||||
.custom-group ha-control-button {
|
.custom-group ha-control-button {
|
||||||
|
@@ -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: 24px;
|
--control-number-buttons-border-radius: 36px;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@@ -186,8 +186,8 @@ export class DemoHaControlSelect extends LitElement {
|
|||||||
.custom {
|
.custom {
|
||||||
--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: 100px;
|
--control-select-thickness: 130px;
|
||||||
--control-select-border-radius: 24px;
|
--control-select-border-radius: 36px;
|
||||||
}
|
}
|
||||||
.vertical-selects {
|
.vertical-selects {
|
||||||
height: 300px;
|
height: 300px;
|
||||||
|
@@ -150,8 +150,8 @@ export class DemoHaBarSlider extends LitElement {
|
|||||||
--control-slider-color: #ffcf4c;
|
--control-slider-color: #ffcf4c;
|
||||||
--control-slider-background: #ffcf4c;
|
--control-slider-background: #ffcf4c;
|
||||||
--control-slider-background-opacity: 0.2;
|
--control-slider-background-opacity: 0.2;
|
||||||
--control-slider-thickness: 100px;
|
--control-slider-thickness: 130px;
|
||||||
--control-slider-border-radius: 24px;
|
--control-slider-border-radius: 36px;
|
||||||
}
|
}
|
||||||
.vertical-sliders {
|
.vertical-sliders {
|
||||||
height: 300px;
|
height: 300px;
|
||||||
|
@@ -117,8 +117,8 @@ export class DemoHaControlSwitch extends LitElement {
|
|||||||
.custom {
|
.custom {
|
||||||
--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: 100px;
|
--control-switch-thickness: 130px;
|
||||||
--control-switch-border-radius: 24px;
|
--control-switch-border-radius: 36px;
|
||||||
--control-switch-padding: 6px;
|
--control-switch-padding: 6px;
|
||||||
--mdc-icon-size: 24px;
|
--mdc-icon-size: 24px;
|
||||||
}
|
}
|
||||||
|
@@ -59,6 +59,7 @@ const DEVICES = [
|
|||||||
hw_version: null,
|
hw_version: null,
|
||||||
via_device_id: null,
|
via_device_id: null,
|
||||||
serial_number: null,
|
serial_number: null,
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
area_id: "backyard",
|
area_id: "backyard",
|
||||||
@@ -77,6 +78,7 @@ const DEVICES = [
|
|||||||
hw_version: null,
|
hw_version: null,
|
||||||
via_device_id: null,
|
via_device_id: null,
|
||||||
serial_number: null,
|
serial_number: null,
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
area_id: null,
|
area_id: null,
|
||||||
@@ -95,30 +97,37 @@ const DEVICES = [
|
|||||||
hw_version: null,
|
hw_version: null,
|
||||||
via_device_id: null,
|
via_device_id: null,
|
||||||
serial_number: null,
|
serial_number: null,
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const AREAS: AreaRegistryEntry[] = [
|
const AREAS: AreaRegistryEntry[] = [
|
||||||
{
|
{
|
||||||
area_id: "backyard",
|
area_id: "backyard",
|
||||||
|
floor_id: null,
|
||||||
name: "Backyard",
|
name: "Backyard",
|
||||||
icon: null,
|
icon: null,
|
||||||
picture: null,
|
picture: null,
|
||||||
aliases: [],
|
aliases: [],
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
area_id: "bedroom",
|
area_id: "bedroom",
|
||||||
|
floor_id: null,
|
||||||
name: "Bedroom",
|
name: "Bedroom",
|
||||||
icon: "mdi:bed",
|
icon: "mdi:bed",
|
||||||
picture: null,
|
picture: null,
|
||||||
aliases: [],
|
aliases: [],
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
area_id: "livingroom",
|
area_id: "livingroom",
|
||||||
|
floor_id: null,
|
||||||
name: "Livingroom",
|
name: "Livingroom",
|
||||||
icon: "mdi:sofa",
|
icon: "mdi:sofa",
|
||||||
picture: null,
|
picture: null,
|
||||||
aliases: [],
|
aliases: [],
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
@@ -17,6 +17,10 @@ import { provideHass } from "../../../../src/fake_data/provide_hass";
|
|||||||
import { ProvideHassElement } from "../../../../src/mixins/provide-hass-lit-mixin";
|
import { ProvideHassElement } from "../../../../src/mixins/provide-hass-lit-mixin";
|
||||||
import type { HomeAssistant } from "../../../../src/types";
|
import type { HomeAssistant } from "../../../../src/types";
|
||||||
import "../../components/demo-black-white-row";
|
import "../../components/demo-black-white-row";
|
||||||
|
import { FloorRegistryEntry } from "../../../../src/data/floor_registry";
|
||||||
|
import { LabelRegistryEntry } from "../../../../src/data/label_registry";
|
||||||
|
import { mockFloorRegistry } from "../../../../demo/src/stubs/floor_registry";
|
||||||
|
import { mockLabelRegistry } from "../../../../demo/src/stubs/label_registry";
|
||||||
|
|
||||||
const ENTITIES = [
|
const ENTITIES = [
|
||||||
getEntity("alarm_control_panel", "alarm", "disarmed", {
|
getEntity("alarm_control_panel", "alarm", "disarmed", {
|
||||||
@@ -55,6 +59,7 @@ const DEVICES = [
|
|||||||
hw_version: null,
|
hw_version: null,
|
||||||
via_device_id: null,
|
via_device_id: null,
|
||||||
serial_number: null,
|
serial_number: null,
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
area_id: "backyard",
|
area_id: "backyard",
|
||||||
@@ -73,6 +78,7 @@ const DEVICES = [
|
|||||||
hw_version: null,
|
hw_version: null,
|
||||||
via_device_id: null,
|
via_device_id: null,
|
||||||
serial_number: null,
|
serial_number: null,
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
area_id: null,
|
area_id: null,
|
||||||
@@ -91,30 +97,78 @@ const DEVICES = [
|
|||||||
hw_version: null,
|
hw_version: null,
|
||||||
via_device_id: null,
|
via_device_id: null,
|
||||||
serial_number: null,
|
serial_number: null,
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const AREAS: AreaRegistryEntry[] = [
|
const AREAS: AreaRegistryEntry[] = [
|
||||||
{
|
{
|
||||||
area_id: "backyard",
|
area_id: "backyard",
|
||||||
|
floor_id: "ground",
|
||||||
name: "Backyard",
|
name: "Backyard",
|
||||||
icon: null,
|
icon: null,
|
||||||
picture: null,
|
picture: null,
|
||||||
aliases: [],
|
aliases: [],
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
area_id: "bedroom",
|
area_id: "bedroom",
|
||||||
|
floor_id: "first",
|
||||||
name: "Bedroom",
|
name: "Bedroom",
|
||||||
icon: "mdi:bed",
|
icon: "mdi:bed",
|
||||||
picture: null,
|
picture: null,
|
||||||
aliases: [],
|
aliases: [],
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
area_id: "livingroom",
|
area_id: "livingroom",
|
||||||
|
floor_id: "ground",
|
||||||
name: "Livingroom",
|
name: "Livingroom",
|
||||||
icon: "mdi:sofa",
|
icon: "mdi:sofa",
|
||||||
picture: null,
|
picture: null,
|
||||||
aliases: [],
|
aliases: [],
|
||||||
|
labels: [],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const FLOORS: FloorRegistryEntry[] = [
|
||||||
|
{
|
||||||
|
floor_id: "ground",
|
||||||
|
name: "Ground floor",
|
||||||
|
level: 0,
|
||||||
|
icon: null,
|
||||||
|
aliases: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
floor_id: "first",
|
||||||
|
name: "First floor",
|
||||||
|
level: 1,
|
||||||
|
icon: "mdi:numeric-1",
|
||||||
|
aliases: [],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
floor_id: "second",
|
||||||
|
name: "Second floor",
|
||||||
|
level: 2,
|
||||||
|
icon: "mdi:numeric-2",
|
||||||
|
aliases: [],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const LABELS: LabelRegistryEntry[] = [
|
||||||
|
{
|
||||||
|
label_id: "energy",
|
||||||
|
name: "Energy",
|
||||||
|
icon: null,
|
||||||
|
color: "yellow",
|
||||||
|
description: null,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label_id: "entertainment",
|
||||||
|
name: "Entertainment",
|
||||||
|
icon: "mdi:popcorn",
|
||||||
|
color: "blue",
|
||||||
|
description: null,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -125,7 +179,12 @@ const SCHEMAS: {
|
|||||||
{
|
{
|
||||||
name: "One of each",
|
name: "One of each",
|
||||||
input: {
|
input: {
|
||||||
|
label: { name: "Label", selector: { label: {} } },
|
||||||
|
floor: { name: "Floor", selector: { floor: {} } },
|
||||||
|
area: { name: "Area", selector: { area: {} } },
|
||||||
|
device: { name: "Device", selector: { device: {} } },
|
||||||
entity: { name: "Entity", selector: { entity: {} } },
|
entity: { name: "Entity", selector: { entity: {} } },
|
||||||
|
target: { name: "Target", selector: { target: {} } },
|
||||||
state: {
|
state: {
|
||||||
name: "State",
|
name: "State",
|
||||||
selector: { state: { entity_id: "alarm_control_panel.alarm" } },
|
selector: { state: { entity_id: "alarm_control_panel.alarm" } },
|
||||||
@@ -134,15 +193,12 @@ const SCHEMAS: {
|
|||||||
name: "Attribute",
|
name: "Attribute",
|
||||||
selector: { attribute: { entity_id: "" } },
|
selector: { attribute: { entity_id: "" } },
|
||||||
},
|
},
|
||||||
device: { name: "Device", selector: { device: {} } },
|
|
||||||
config_entry: {
|
config_entry: {
|
||||||
name: "Integration",
|
name: "Integration",
|
||||||
selector: { config_entry: {} },
|
selector: { config_entry: {} },
|
||||||
},
|
},
|
||||||
duration: { name: "Duration", selector: { duration: {} } },
|
duration: { name: "Duration", selector: { duration: {} } },
|
||||||
addon: { name: "Addon", selector: { addon: {} } },
|
addon: { name: "Addon", selector: { addon: {} } },
|
||||||
area: { name: "Area", selector: { area: {} } },
|
|
||||||
target: { name: "Target", selector: { target: {} } },
|
|
||||||
number_box: {
|
number_box: {
|
||||||
name: "Number Box",
|
name: "Number Box",
|
||||||
selector: {
|
selector: {
|
||||||
@@ -291,6 +347,8 @@ const SCHEMAS: {
|
|||||||
entity: { name: "Entity", selector: { entity: { multiple: true } } },
|
entity: { name: "Entity", selector: { entity: { multiple: true } } },
|
||||||
device: { name: "Device", selector: { device: { multiple: true } } },
|
device: { name: "Device", selector: { device: { multiple: true } } },
|
||||||
area: { name: "Area", selector: { area: { multiple: true } } },
|
area: { name: "Area", selector: { area: { multiple: true } } },
|
||||||
|
floor: { name: "Floor", selector: { floor: { multiple: true } } },
|
||||||
|
label: { name: "Label", selector: { label: { multiple: true } } },
|
||||||
select: {
|
select: {
|
||||||
name: "Select Multiple",
|
name: "Select Multiple",
|
||||||
selector: {
|
selector: {
|
||||||
@@ -347,6 +405,8 @@ class DemoHaSelector extends LitElement implements ProvideHassElement {
|
|||||||
mockDeviceRegistry(hass, DEVICES);
|
mockDeviceRegistry(hass, DEVICES);
|
||||||
mockConfigEntries(hass);
|
mockConfigEntries(hass);
|
||||||
mockAreaRegistry(hass, AREAS);
|
mockAreaRegistry(hass, AREAS);
|
||||||
|
mockFloorRegistry(hass, FLOORS);
|
||||||
|
mockLabelRegistry(hass, LABELS);
|
||||||
mockHassioSupervisor(hass);
|
mockHassioSupervisor(hass);
|
||||||
hass.mockWS("auth/sign_path", (params) => params);
|
hass.mockWS("auth/sign_path", (params) => params);
|
||||||
hass.mockWS("media_player/browse_media", this._browseMedia);
|
hass.mockWS("media_player/browse_media", this._browseMedia);
|
||||||
|
@@ -11,7 +11,7 @@ const ENTITIES = [
|
|||||||
latitude: 32.877105,
|
latitude: 32.877105,
|
||||||
longitude: 117.232185,
|
longitude: 117.232185,
|
||||||
gps_accuracy: 91,
|
gps_accuracy: 91,
|
||||||
battery: 71,
|
battery: 25,
|
||||||
friendly_name: "Paulus",
|
friendly_name: "Paulus",
|
||||||
}),
|
}),
|
||||||
getEntity("device_tracker", "demo_anne_therese", "school", {
|
getEntity("device_tracker", "demo_anne_therese", "school", {
|
||||||
@@ -19,7 +19,7 @@ const ENTITIES = [
|
|||||||
latitude: 32.877105,
|
latitude: 32.877105,
|
||||||
longitude: 117.232185,
|
longitude: 117.232185,
|
||||||
gps_accuracy: 91,
|
gps_accuracy: 91,
|
||||||
battery: 71,
|
battery: 50,
|
||||||
friendly_name: "Anne Therese",
|
friendly_name: "Anne Therese",
|
||||||
}),
|
}),
|
||||||
getEntity("device_tracker", "demo_home_boy", "home", {
|
getEntity("device_tracker", "demo_home_boy", "home", {
|
||||||
@@ -27,7 +27,7 @@ const ENTITIES = [
|
|||||||
latitude: 32.877105,
|
latitude: 32.877105,
|
||||||
longitude: 117.232185,
|
longitude: 117.232185,
|
||||||
gps_accuracy: 91,
|
gps_accuracy: 91,
|
||||||
battery: 71,
|
battery: 75,
|
||||||
friendly_name: "Home Boy",
|
friendly_name: "Home Boy",
|
||||||
}),
|
}),
|
||||||
getEntity("light", "bed_light", "on", {
|
getEntity("light", "bed_light", "on", {
|
||||||
@@ -39,21 +39,53 @@ const ENTITIES = [
|
|||||||
getEntity("light", "ceiling_lights", "off", {
|
getEntity("light", "ceiling_lights", "off", {
|
||||||
friendly_name: "Ceiling Lights",
|
friendly_name: "Ceiling Lights",
|
||||||
}),
|
}),
|
||||||
|
getEntity("sensor", "battery_1", 20, {
|
||||||
|
device_class: "battery",
|
||||||
|
friendly_name: "Battery 1",
|
||||||
|
unit_of_measurement: "%",
|
||||||
|
}),
|
||||||
|
getEntity("sensor", "battery_2", 35, {
|
||||||
|
device_class: "battery",
|
||||||
|
friendly_name: "Battery 2",
|
||||||
|
unit_of_measurement: "%",
|
||||||
|
}),
|
||||||
|
getEntity("sensor", "battery_3", 40, {
|
||||||
|
device_class: "battery",
|
||||||
|
friendly_name: "Battery 3",
|
||||||
|
unit_of_measurement: "%",
|
||||||
|
}),
|
||||||
|
getEntity("sensor", "battery_4", 80, {
|
||||||
|
device_class: "battery",
|
||||||
|
friendly_name: "Battery 4",
|
||||||
|
unit_of_measurement: "%",
|
||||||
|
}),
|
||||||
|
getEntity("input_number", "min_battery_level", 30, {
|
||||||
|
mode: "slider",
|
||||||
|
step: 10,
|
||||||
|
min: 0,
|
||||||
|
max: 100,
|
||||||
|
icon: "mdi:battery-alert-variant",
|
||||||
|
friendly_name: "Minimum Battery Level",
|
||||||
|
unit_of_measurement: "%",
|
||||||
|
}),
|
||||||
];
|
];
|
||||||
|
|
||||||
const CONFIGS = [
|
const CONFIGS = [
|
||||||
{
|
{
|
||||||
heading: "Unfiltered controller",
|
heading: "Unfiltered entities",
|
||||||
config: `
|
config: `
|
||||||
- type: entities
|
- type: entities
|
||||||
entities:
|
entities:
|
||||||
- light.bed_light
|
- device_tracker.demo_anne_therese
|
||||||
- light.ceiling_lights
|
- device_tracker.demo_home_boy
|
||||||
- light.kitchen_lights
|
- device_tracker.demo_paulus
|
||||||
|
- light.bed_light
|
||||||
|
- light.ceiling_lights
|
||||||
|
- light.kitchen_lights
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
heading: "Filtered entities card",
|
heading: "On and home entities",
|
||||||
config: `
|
config: `
|
||||||
- type: entity-filter
|
- type: entity-filter
|
||||||
entities:
|
entities:
|
||||||
@@ -63,9 +95,28 @@ const CONFIGS = [
|
|||||||
- light.bed_light
|
- light.bed_light
|
||||||
- light.ceiling_lights
|
- light.ceiling_lights
|
||||||
- light.kitchen_lights
|
- light.kitchen_lights
|
||||||
state_filter:
|
conditions:
|
||||||
- "on"
|
- condition: state
|
||||||
- home
|
state:
|
||||||
|
- "on"
|
||||||
|
- home
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Same state as Bed Light",
|
||||||
|
config: `
|
||||||
|
- type: entity-filter
|
||||||
|
entities:
|
||||||
|
- device_tracker.demo_anne_therese
|
||||||
|
- device_tracker.demo_home_boy
|
||||||
|
- device_tracker.demo_paulus
|
||||||
|
- light.bed_light
|
||||||
|
- light.ceiling_lights
|
||||||
|
- light.kitchen_lights
|
||||||
|
conditions:
|
||||||
|
- condition: state
|
||||||
|
state:
|
||||||
|
- light.bed_light
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -79,9 +130,11 @@ const CONFIGS = [
|
|||||||
- light.bed_light
|
- light.bed_light
|
||||||
- light.ceiling_lights
|
- light.ceiling_lights
|
||||||
- light.kitchen_lights
|
- light.kitchen_lights
|
||||||
state_filter:
|
conditions:
|
||||||
- "on"
|
- condition: state
|
||||||
- not_home
|
state:
|
||||||
|
- "on"
|
||||||
|
- home
|
||||||
card:
|
card:
|
||||||
type: entities
|
type: entities
|
||||||
title: Custom Title
|
title: Custom Title
|
||||||
@@ -99,15 +152,101 @@ const CONFIGS = [
|
|||||||
- light.bed_light
|
- light.bed_light
|
||||||
- light.ceiling_lights
|
- light.ceiling_lights
|
||||||
- light.kitchen_lights
|
- light.kitchen_lights
|
||||||
state_filter:
|
conditions:
|
||||||
- "on"
|
- condition: state
|
||||||
- not_home
|
state:
|
||||||
|
- "on"
|
||||||
|
- home
|
||||||
card:
|
card:
|
||||||
type: glance
|
type: glance
|
||||||
show_state: true
|
show_state: true
|
||||||
title: Custom Title
|
title: Custom Title
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
heading:
|
||||||
|
"Filtered entities by battery attribute (< '30') using state filter",
|
||||||
|
config: `
|
||||||
|
- type: entity-filter
|
||||||
|
entities:
|
||||||
|
- device_tracker.demo_anne_therese
|
||||||
|
- device_tracker.demo_home_boy
|
||||||
|
- device_tracker.demo_paulus
|
||||||
|
state_filter:
|
||||||
|
- operator: <
|
||||||
|
attribute: battery
|
||||||
|
value: "30"
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Unfiltered number entities",
|
||||||
|
config: `
|
||||||
|
- type: entities
|
||||||
|
entities:
|
||||||
|
- input_number.min_battery_level
|
||||||
|
- sensor.battery_1
|
||||||
|
- sensor.battery_3
|
||||||
|
- sensor.battery_2
|
||||||
|
- sensor.battery_4
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Battery lower than 50%",
|
||||||
|
config: `
|
||||||
|
- type: entity-filter
|
||||||
|
entities:
|
||||||
|
- sensor.battery_1
|
||||||
|
- sensor.battery_3
|
||||||
|
- sensor.battery_2
|
||||||
|
- sensor.battery_4
|
||||||
|
conditions:
|
||||||
|
- condition: numeric_state
|
||||||
|
below: 50
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Battery lower than min battery level",
|
||||||
|
config: `
|
||||||
|
- type: entity-filter
|
||||||
|
entities:
|
||||||
|
- sensor.battery_1
|
||||||
|
- sensor.battery_3
|
||||||
|
- sensor.battery_2
|
||||||
|
- sensor.battery_4
|
||||||
|
conditions:
|
||||||
|
- condition: numeric_state
|
||||||
|
below: input_number.min_battery_level
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Battery between min battery level and 70%",
|
||||||
|
config: `
|
||||||
|
- type: entity-filter
|
||||||
|
entities:
|
||||||
|
- sensor.battery_1
|
||||||
|
- sensor.battery_3
|
||||||
|
- sensor.battery_2
|
||||||
|
- sensor.battery_4
|
||||||
|
conditions:
|
||||||
|
- condition: numeric_state
|
||||||
|
above: input_number.min_battery_level
|
||||||
|
below: 70
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Error: Entities must be specified",
|
||||||
|
config: `
|
||||||
|
- type: entity-filter
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Error: Incorrect filter config",
|
||||||
|
config: `
|
||||||
|
- type: entity-filter
|
||||||
|
entities:
|
||||||
|
- sensor.gas_station_lowest_price
|
||||||
|
`,
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@customElement("demo-lovelace-entity-filter-card")
|
@customElement("demo-lovelace-entity-filter-card")
|
||||||
|
@@ -36,6 +36,45 @@ const ENTITIES = [
|
|||||||
friendly_name: "Nest",
|
friendly_name: "Nest",
|
||||||
supported_features: 43,
|
supported_features: 43,
|
||||||
}),
|
}),
|
||||||
|
getEntity("climate", "overkiz_radiator", "heat", {
|
||||||
|
current_temperature: 18,
|
||||||
|
min_temp: 7,
|
||||||
|
max_temp: 35,
|
||||||
|
temperature: 20,
|
||||||
|
hvac_modes: ["heat", "auto", "off"],
|
||||||
|
friendly_name: "Overkiz radiator",
|
||||||
|
supported_features: 17,
|
||||||
|
preset_mode: "comfort",
|
||||||
|
preset_modes: [
|
||||||
|
"none",
|
||||||
|
"frost_protection",
|
||||||
|
"eco",
|
||||||
|
"comfort",
|
||||||
|
"comfort-1",
|
||||||
|
"comfort-2",
|
||||||
|
"auto",
|
||||||
|
"boost",
|
||||||
|
"external",
|
||||||
|
"prog",
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
getEntity("climate", "overkiz_towel_dryer", "heat", {
|
||||||
|
current_temperature: null,
|
||||||
|
min_temp: 7,
|
||||||
|
max_temp: 35,
|
||||||
|
hvac_modes: ["heat", "off"],
|
||||||
|
friendly_name: "Overkiz towel dryer",
|
||||||
|
supported_features: 16,
|
||||||
|
preset_mode: "eco",
|
||||||
|
preset_modes: [
|
||||||
|
"none",
|
||||||
|
"frost_protection",
|
||||||
|
"eco",
|
||||||
|
"comfort",
|
||||||
|
"comfort-1",
|
||||||
|
"comfort-2",
|
||||||
|
],
|
||||||
|
}),
|
||||||
getEntity("climate", "sensibo", "fan_only", {
|
getEntity("climate", "sensibo", "fan_only", {
|
||||||
current_temperature: null,
|
current_temperature: null,
|
||||||
temperature: null,
|
temperature: null,
|
||||||
@@ -46,7 +85,9 @@ const ENTITIES = [
|
|||||||
friendly_name: "Sensibo purifier",
|
friendly_name: "Sensibo purifier",
|
||||||
fan_modes: ["low", "high"],
|
fan_modes: ["low", "high"],
|
||||||
fan_mode: "low",
|
fan_mode: "low",
|
||||||
supported_features: 9,
|
swing_modes: ["on", "off", "both", "vertical", "horizontal"],
|
||||||
|
swing_mode: "vertical",
|
||||||
|
supported_features: 41,
|
||||||
}),
|
}),
|
||||||
getEntity("climate", "unavailable", "unavailable", {
|
getEntity("climate", "unavailable", "unavailable", {
|
||||||
supported_features: 43,
|
supported_features: 43,
|
||||||
@@ -59,8 +100,6 @@ const CONFIGS = [
|
|||||||
config: `
|
config: `
|
||||||
- type: thermostat
|
- type: thermostat
|
||||||
entity: climate.ecobee
|
entity: climate.ecobee
|
||||||
- type: thermostat
|
|
||||||
entity: climate.nest
|
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -70,6 +109,66 @@ const CONFIGS = [
|
|||||||
entity: climate.nest
|
entity: climate.nest
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
heading: "Feature example",
|
||||||
|
config: `
|
||||||
|
- type: thermostat
|
||||||
|
entity: climate.overkiz_radiator
|
||||||
|
features:
|
||||||
|
- type: climate-hvac-modes
|
||||||
|
hvac_modes:
|
||||||
|
- heat
|
||||||
|
- 'off'
|
||||||
|
- auto
|
||||||
|
- type: climate-preset-modes
|
||||||
|
style: icons
|
||||||
|
preset_modes:
|
||||||
|
- none
|
||||||
|
- frost_protection
|
||||||
|
- eco
|
||||||
|
- comfort
|
||||||
|
- comfort-1
|
||||||
|
- comfort-2
|
||||||
|
- auto
|
||||||
|
- boost
|
||||||
|
- external
|
||||||
|
- prog
|
||||||
|
- type: climate-preset-modes
|
||||||
|
style: dropdown
|
||||||
|
preset_modes:
|
||||||
|
- none
|
||||||
|
- frost_protection
|
||||||
|
- eco
|
||||||
|
- comfort
|
||||||
|
- comfort-1
|
||||||
|
- comfort-2
|
||||||
|
- auto
|
||||||
|
- boost
|
||||||
|
- external
|
||||||
|
- prog
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Preset only example",
|
||||||
|
config: `
|
||||||
|
- type: thermostat
|
||||||
|
entity: climate.overkiz_towel_dryer
|
||||||
|
features:
|
||||||
|
- type: climate-hvac-modes
|
||||||
|
hvac_modes:
|
||||||
|
- heat
|
||||||
|
- 'off'
|
||||||
|
- type: climate-preset-modes
|
||||||
|
style: icons
|
||||||
|
preset_modes:
|
||||||
|
- none
|
||||||
|
- frost_protection
|
||||||
|
- eco
|
||||||
|
- comfort
|
||||||
|
- comfort-1
|
||||||
|
- comfort-2
|
||||||
|
`,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
heading: "Fan only example",
|
heading: "Fan only example",
|
||||||
config: `
|
config: `
|
||||||
@@ -85,6 +184,14 @@ const CONFIGS = [
|
|||||||
fan_modes:
|
fan_modes:
|
||||||
- low
|
- low
|
||||||
- high
|
- high
|
||||||
|
- type: climate-swing-modes
|
||||||
|
style: icons
|
||||||
|
swing_modes:
|
||||||
|
- 'on'
|
||||||
|
- 'off'
|
||||||
|
- 'both'
|
||||||
|
- 'vertical'
|
||||||
|
- 'horizontal'
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@@ -2,6 +2,7 @@ import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
|||||||
import { customElement, query } from "lit/decorators";
|
import { customElement, query } from "lit/decorators";
|
||||||
import { CoverEntityFeature } from "../../../../src/data/cover";
|
import { CoverEntityFeature } from "../../../../src/data/cover";
|
||||||
import { LightColorMode } from "../../../../src/data/light";
|
import { LightColorMode } from "../../../../src/data/light";
|
||||||
|
import { LockEntityFeature } from "../../../../src/data/lock";
|
||||||
import { VacuumEntityFeature } from "../../../../src/data/vacuum";
|
import { VacuumEntityFeature } from "../../../../src/data/vacuum";
|
||||||
import { getEntity } from "../../../../src/fake_data/entity";
|
import { getEntity } from "../../../../src/fake_data/entity";
|
||||||
import { provideHass } from "../../../../src/fake_data/provide_hass";
|
import { provideHass } from "../../../../src/fake_data/provide_hass";
|
||||||
@@ -20,6 +21,11 @@ const ENTITIES = [
|
|||||||
getEntity("light", "unavailable", "unavailable", {
|
getEntity("light", "unavailable", "unavailable", {
|
||||||
friendly_name: "Unavailable entity",
|
friendly_name: "Unavailable entity",
|
||||||
}),
|
}),
|
||||||
|
getEntity("lock", "front_door", "locked", {
|
||||||
|
friendly_name: "Front Door Lock",
|
||||||
|
device_class: "lock",
|
||||||
|
supported_features: LockEntityFeature.OPEN,
|
||||||
|
}),
|
||||||
getEntity("climate", "thermostat", "heat", {
|
getEntity("climate", "thermostat", "heat", {
|
||||||
current_temperature: 73,
|
current_temperature: 73,
|
||||||
min_temp: 45,
|
min_temp: 45,
|
||||||
@@ -138,6 +144,24 @@ const CONFIGS = [
|
|||||||
- type: "color-temp"
|
- type: "color-temp"
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
heading: "Lock commands feature",
|
||||||
|
config: `
|
||||||
|
- type: tile
|
||||||
|
entity: lock.front_door
|
||||||
|
features:
|
||||||
|
- type: "lock-commands"
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Lock open door feature",
|
||||||
|
config: `
|
||||||
|
- type: tile
|
||||||
|
entity: lock.front_door
|
||||||
|
features:
|
||||||
|
- type: "lock-open-door"
|
||||||
|
`,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
heading: "Vacuum commands feature",
|
heading: "Vacuum commands feature",
|
||||||
config: `
|
config: `
|
||||||
|
@@ -406,6 +406,7 @@ export class DemoEntityState extends LitElement {
|
|||||||
entity_id: "select.speed",
|
entity_id: "select.speed",
|
||||||
translation_key: "speed",
|
translation_key: "speed",
|
||||||
platform: "demo",
|
platform: "demo",
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@@ -31,10 +31,13 @@ const createConfigEntry = (
|
|||||||
supports_options: false,
|
supports_options: false,
|
||||||
supports_remove_device: false,
|
supports_remove_device: false,
|
||||||
supports_unload: true,
|
supports_unload: true,
|
||||||
|
supports_reconfigure: true,
|
||||||
disabled_by: null,
|
disabled_by: null,
|
||||||
pref_disable_new_entities: false,
|
pref_disable_new_entities: false,
|
||||||
pref_disable_polling: false,
|
pref_disable_polling: false,
|
||||||
reason: null,
|
reason: null,
|
||||||
|
error_reason_translation_key: null,
|
||||||
|
error_reason_translation_placeholders: null,
|
||||||
...override,
|
...override,
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -198,6 +201,8 @@ const createEntityRegistryEntries = (
|
|||||||
has_entity_name: false,
|
has_entity_name: false,
|
||||||
unique_id: "updater",
|
unique_id: "updater",
|
||||||
options: null,
|
options: null,
|
||||||
|
labels: [],
|
||||||
|
categories: {},
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -221,6 +226,7 @@ const createDeviceRegistryEntries = (
|
|||||||
name_by_user: null,
|
name_by_user: null,
|
||||||
disabled_by: null,
|
disabled_by: null,
|
||||||
configuration_url: null,
|
configuration_url: null,
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
@@ -11,7 +11,7 @@ import "../../components/demo-more-infos";
|
|||||||
import { ClimateEntityFeature } from "../../../../src/data/climate";
|
import { ClimateEntityFeature } from "../../../../src/data/climate";
|
||||||
|
|
||||||
const ENTITIES = [
|
const ENTITIES = [
|
||||||
getEntity("climate", "thermostat", "heat", {
|
getEntity("climate", "radiator", "heat", {
|
||||||
friendly_name: "Basic heater",
|
friendly_name: "Basic heater",
|
||||||
hvac_modes: ["heat", "off"],
|
hvac_modes: ["heat", "off"],
|
||||||
hvac_mode: "heat",
|
hvac_mode: "heat",
|
||||||
@@ -80,6 +80,24 @@ const ENTITIES = [
|
|||||||
max_humidity: 100,
|
max_humidity: 100,
|
||||||
humidity: 50,
|
humidity: 50,
|
||||||
}),
|
}),
|
||||||
|
getEntity("climate", "towel_dryer", "heat", {
|
||||||
|
friendly_name: "Preset only heater",
|
||||||
|
hvac_modes: ["heat", "off"],
|
||||||
|
hvac_mode: "heat",
|
||||||
|
preset_modes: [
|
||||||
|
"none",
|
||||||
|
"frost_protection",
|
||||||
|
"eco",
|
||||||
|
"comfort",
|
||||||
|
"comfort-1",
|
||||||
|
"comfort-2",
|
||||||
|
],
|
||||||
|
preset_mode: "eco",
|
||||||
|
current_temperature: null,
|
||||||
|
min_temp: 7,
|
||||||
|
max_temp: 35,
|
||||||
|
supported_features: ClimateEntityFeature.PRESET_MODE,
|
||||||
|
}),
|
||||||
getEntity("climate", "unavailable", "unavailable", {
|
getEntity("climate", "unavailable", "unavailable", {
|
||||||
friendly_name: "Unavailable heater",
|
friendly_name: "Unavailable heater",
|
||||||
hvac_modes: ["heat", "off"],
|
hvac_modes: ["heat", "off"],
|
||||||
|
@@ -1,4 +1,7 @@
|
|||||||
import { globIterate } from "glob";
|
import { globIterate } from "glob";
|
||||||
|
import { availableParallelism } from "node:os";
|
||||||
|
|
||||||
|
process.env.UV_THREADPOOL_SIZE = availableParallelism();
|
||||||
|
|
||||||
const gulpImports = [];
|
const gulpImports = [];
|
||||||
|
|
||||||
|
@@ -1263,6 +1263,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
.card-actions {
|
.card-actions {
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
direction: var(--direction);
|
||||||
}
|
}
|
||||||
.changelog {
|
.changelog {
|
||||||
display: contents;
|
display: contents;
|
||||||
|
@@ -154,12 +154,16 @@ class HassioHardwareDialog extends LitElement {
|
|||||||
ha-icon-button {
|
ha-icon-button {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
right: 16px;
|
right: 16px;
|
||||||
|
inset-inline-end: 16px;
|
||||||
|
inset-inline-start: initial;
|
||||||
top: 10px;
|
top: 10px;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
color: var(--primary-text-color);
|
color: var(--primary-text-color);
|
||||||
}
|
}
|
||||||
h2 {
|
h2 {
|
||||||
margin: 18px 42px 0 18px;
|
margin: 18px 42px 0 18px;
|
||||||
|
margin-inline-start: 18px;
|
||||||
|
margin-inline-end: 42px;
|
||||||
color: var(--primary-text-color);
|
color: var(--primary-text-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,7 +1,5 @@
|
|||||||
import "@material/mwc-button/mwc-button";
|
import "@material/mwc-button/mwc-button";
|
||||||
import { mdiDelete, mdiDeleteOff } from "@mdi/js";
|
import { mdiDelete, mdiDeleteOff } from "@mdi/js";
|
||||||
import "@polymer/paper-item/paper-item";
|
|
||||||
import "@polymer/paper-item/paper-item-body";
|
|
||||||
import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
|
import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
|
||||||
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
@@ -27,6 +25,8 @@ import type { HomeAssistant } from "../../../../src/types";
|
|||||||
import { HassioRepositoryDialogParams } from "./show-dialog-repositories";
|
import { HassioRepositoryDialogParams } from "./show-dialog-repositories";
|
||||||
import type { HaTextField } from "../../../../src/components/ha-textfield";
|
import type { HaTextField } from "../../../../src/components/ha-textfield";
|
||||||
import "../../../../src/components/ha-textfield";
|
import "../../../../src/components/ha-textfield";
|
||||||
|
import "../../../../src/components/ha-list-new";
|
||||||
|
import "../../../../src/components/ha-list-item-new";
|
||||||
|
|
||||||
@customElement("dialog-hassio-repositories")
|
@customElement("dialog-hassio-repositories")
|
||||||
class HassioRepositoriesDialog extends LitElement {
|
class HassioRepositoriesDialog extends LitElement {
|
||||||
@@ -106,44 +106,46 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
? html`<ha-alert alert-type="error">${this._error}</ha-alert>`
|
? html`<ha-alert alert-type="error">${this._error}</ha-alert>`
|
||||||
: ""}
|
: ""}
|
||||||
<div class="form">
|
<div class="form">
|
||||||
${repositories.length
|
<ha-list-new>
|
||||||
? repositories.map(
|
${repositories.length
|
||||||
(repo) => html`
|
? repositories.map(
|
||||||
<paper-item class="option">
|
(repo) => html`
|
||||||
<paper-item-body three-line>
|
<ha-list-item-new class="option">
|
||||||
<div>${repo.name}</div>
|
${repo.name}
|
||||||
<div secondary>${repo.maintainer}</div>
|
<div slot="supporting-text">
|
||||||
<div secondary>${repo.url}</div>
|
<div>${repo.maintainer}</div>
|
||||||
</paper-item-body>
|
<div>${repo.url}</div>
|
||||||
<div class="delete">
|
</div>
|
||||||
<ha-icon-button
|
<div class="delete" slot="end">
|
||||||
.label=${this._dialogParams!.supervisor.localize(
|
<ha-icon-button
|
||||||
"dialog.repositories.remove"
|
.label=${this._dialogParams!.supervisor.localize(
|
||||||
)}
|
"dialog.repositories.remove"
|
||||||
.disabled=${usedRepositories.includes(repo.slug)}
|
)}
|
||||||
.slug=${repo.slug}
|
.disabled=${usedRepositories.includes(repo.slug)}
|
||||||
.path=${usedRepositories.includes(repo.slug)
|
.slug=${repo.slug}
|
||||||
? mdiDeleteOff
|
.path=${usedRepositories.includes(repo.slug)
|
||||||
: mdiDelete}
|
? mdiDeleteOff
|
||||||
@click=${this._removeRepository}
|
: mdiDelete}
|
||||||
>
|
@click=${this._removeRepository}
|
||||||
</ha-icon-button>
|
>
|
||||||
<simple-tooltip
|
</ha-icon-button>
|
||||||
animation-delay="0"
|
<simple-tooltip
|
||||||
position="bottom"
|
animation-delay="0"
|
||||||
offset="1"
|
position="bottom"
|
||||||
>
|
offset="1"
|
||||||
${this._dialogParams!.supervisor.localize(
|
>
|
||||||
usedRepositories.includes(repo.slug)
|
${this._dialogParams!.supervisor.localize(
|
||||||
? "dialog.repositories.used"
|
usedRepositories.includes(repo.slug)
|
||||||
: "dialog.repositories.remove"
|
? "dialog.repositories.used"
|
||||||
)}
|
: "dialog.repositories.remove"
|
||||||
</simple-tooltip>
|
)}
|
||||||
</div>
|
</simple-tooltip>
|
||||||
</paper-item>
|
</div>
|
||||||
`
|
</ha-list-item-new>
|
||||||
)
|
`
|
||||||
: html`<paper-item> No repositories </paper-item>`}
|
)
|
||||||
|
: html`<ha-list-item-new> No repositories </ha-list-item-new>`}
|
||||||
|
</ha-list-new>
|
||||||
<div class="layout horizontal bottom">
|
<div class="layout horizontal bottom">
|
||||||
<ha-textfield
|
<ha-textfield
|
||||||
class="flex-auto"
|
class="flex-auto"
|
||||||
@@ -206,6 +208,9 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
div.delete ha-icon-button {
|
div.delete ha-icon-button {
|
||||||
color: var(--error-color);
|
color: var(--error-color);
|
||||||
}
|
}
|
||||||
|
ha-list-item-new {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
138
package.json
138
package.json
@@ -25,36 +25,36 @@
|
|||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@babel/runtime": "7.23.9",
|
"@babel/runtime": "7.24.4",
|
||||||
"@braintree/sanitize-url": "7.0.0",
|
"@braintree/sanitize-url": "7.0.1",
|
||||||
"@codemirror/autocomplete": "6.12.0",
|
"@codemirror/autocomplete": "6.16.0",
|
||||||
"@codemirror/commands": "6.3.3",
|
"@codemirror/commands": "6.5.0",
|
||||||
"@codemirror/language": "6.10.1",
|
"@codemirror/language": "6.10.1",
|
||||||
"@codemirror/legacy-modes": "6.3.3",
|
"@codemirror/legacy-modes": "6.4.0",
|
||||||
"@codemirror/search": "6.5.6",
|
"@codemirror/search": "6.5.6",
|
||||||
"@codemirror/state": "6.4.0",
|
"@codemirror/state": "6.4.1",
|
||||||
"@codemirror/view": "6.24.0",
|
"@codemirror/view": "6.26.3",
|
||||||
"@egjs/hammerjs": "2.0.17",
|
"@egjs/hammerjs": "2.0.17",
|
||||||
"@formatjs/intl-datetimeformat": "6.12.2",
|
"@formatjs/intl-datetimeformat": "6.12.3",
|
||||||
"@formatjs/intl-displaynames": "6.6.6",
|
"@formatjs/intl-displaynames": "6.6.6",
|
||||||
"@formatjs/intl-getcanonicallocales": "2.3.0",
|
"@formatjs/intl-getcanonicallocales": "2.3.0",
|
||||||
"@formatjs/intl-listformat": "7.5.5",
|
"@formatjs/intl-listformat": "7.5.5",
|
||||||
"@formatjs/intl-locale": "3.4.5",
|
"@formatjs/intl-locale": "3.4.5",
|
||||||
"@formatjs/intl-numberformat": "8.10.0",
|
"@formatjs/intl-numberformat": "8.10.1",
|
||||||
"@formatjs/intl-pluralrules": "5.2.12",
|
"@formatjs/intl-pluralrules": "5.2.12",
|
||||||
"@formatjs/intl-relativetimeformat": "11.2.12",
|
"@formatjs/intl-relativetimeformat": "11.2.12",
|
||||||
"@fullcalendar/core": "6.1.10",
|
"@fullcalendar/core": "6.1.11",
|
||||||
"@fullcalendar/daygrid": "6.1.10",
|
"@fullcalendar/daygrid": "6.1.11",
|
||||||
"@fullcalendar/interaction": "6.1.10",
|
"@fullcalendar/interaction": "6.1.11",
|
||||||
"@fullcalendar/list": "6.1.10",
|
"@fullcalendar/list": "6.1.11",
|
||||||
"@fullcalendar/luxon3": "6.1.10",
|
"@fullcalendar/luxon3": "6.1.11",
|
||||||
"@fullcalendar/timegrid": "6.1.10",
|
"@fullcalendar/timegrid": "6.1.11",
|
||||||
"@lezer/highlight": "1.2.0",
|
"@lezer/highlight": "1.2.0",
|
||||||
"@lit-labs/context": "0.4.1",
|
"@lit-labs/context": "0.4.1",
|
||||||
"@lit-labs/motion": "1.0.7",
|
"@lit-labs/motion": "1.0.7",
|
||||||
"@lit-labs/observers": "2.0.2",
|
"@lit-labs/observers": "2.0.2",
|
||||||
"@lit-labs/virtualizer": "2.0.12",
|
"@lit-labs/virtualizer": "2.0.12",
|
||||||
"@lrnwebcomponents/simple-tooltip": "patch:@lrnwebcomponents/simple-tooltip@npm%3A8.0.0#~/.yarn/patches/@lrnwebcomponents-simple-tooltip-npm-8.0.0-77591f2e0c.patch",
|
"@lrnwebcomponents/simple-tooltip": "8.0.2",
|
||||||
"@material/chips": "=14.0.0-canary.53b3cad2f.0",
|
"@material/chips": "=14.0.0-canary.53b3cad2f.0",
|
||||||
"@material/data-table": "=14.0.0-canary.53b3cad2f.0",
|
"@material/data-table": "=14.0.0-canary.53b3cad2f.0",
|
||||||
"@material/mwc-base": "0.27.0",
|
"@material/mwc-base": "0.27.0",
|
||||||
@@ -72,6 +72,7 @@
|
|||||||
"@material/mwc-radio": "0.27.0",
|
"@material/mwc-radio": "0.27.0",
|
||||||
"@material/mwc-ripple": "0.27.0",
|
"@material/mwc-ripple": "0.27.0",
|
||||||
"@material/mwc-select": "0.27.0",
|
"@material/mwc-select": "0.27.0",
|
||||||
|
"@material/mwc-snackbar": "0.27.0",
|
||||||
"@material/mwc-switch": "0.27.0",
|
"@material/mwc-switch": "0.27.0",
|
||||||
"@material/mwc-tab": "0.27.0",
|
"@material/mwc-tab": "0.27.0",
|
||||||
"@material/mwc-tab-bar": "0.27.0",
|
"@material/mwc-tab-bar": "0.27.0",
|
||||||
@@ -80,17 +81,16 @@
|
|||||||
"@material/mwc-top-app-bar": "0.27.0",
|
"@material/mwc-top-app-bar": "0.27.0",
|
||||||
"@material/mwc-top-app-bar-fixed": "0.27.0",
|
"@material/mwc-top-app-bar-fixed": "0.27.0",
|
||||||
"@material/top-app-bar": "=14.0.0-canary.53b3cad2f.0",
|
"@material/top-app-bar": "=14.0.0-canary.53b3cad2f.0",
|
||||||
"@material/web": "=1.2.0",
|
"@material/web": "1.4.1",
|
||||||
"@mdi/js": "7.4.47",
|
"@mdi/js": "7.4.47",
|
||||||
"@mdi/svg": "7.4.47",
|
"@mdi/svg": "7.4.47",
|
||||||
"@polymer/paper-item": "3.0.1",
|
"@polymer/paper-item": "3.0.1",
|
||||||
"@polymer/paper-listbox": "3.0.1",
|
"@polymer/paper-listbox": "3.0.1",
|
||||||
"@polymer/paper-tabs": "3.1.0",
|
"@polymer/paper-tabs": "3.1.0",
|
||||||
"@polymer/paper-toast": "3.0.1",
|
|
||||||
"@polymer/polymer": "3.5.1",
|
"@polymer/polymer": "3.5.1",
|
||||||
"@thomasloven/round-slider": "0.6.0",
|
"@thomasloven/round-slider": "0.6.0",
|
||||||
"@vaadin/combo-box": "24.3.6",
|
"@vaadin/combo-box": "24.3.11",
|
||||||
"@vaadin/vaadin-themable-mixin": "24.3.6",
|
"@vaadin/vaadin-themable-mixin": "24.3.11",
|
||||||
"@vibrant/color": "3.2.1-alpha.1",
|
"@vibrant/color": "3.2.1-alpha.1",
|
||||||
"@vibrant/core": "3.2.1-alpha.1",
|
"@vibrant/core": "3.2.1-alpha.1",
|
||||||
"@vibrant/quantizer-mmcq": "3.2.1-alpha.1",
|
"@vibrant/quantizer-mmcq": "3.2.1-alpha.1",
|
||||||
@@ -98,19 +98,20 @@
|
|||||||
"@webcomponents/scoped-custom-element-registry": "0.0.9",
|
"@webcomponents/scoped-custom-element-registry": "0.0.9",
|
||||||
"@webcomponents/webcomponentsjs": "2.8.0",
|
"@webcomponents/webcomponentsjs": "2.8.0",
|
||||||
"app-datepicker": "5.1.1",
|
"app-datepicker": "5.1.1",
|
||||||
"chart.js": "4.4.1",
|
"chart.js": "4.4.2",
|
||||||
|
"color-name": "2.0.0",
|
||||||
"comlink": "4.4.1",
|
"comlink": "4.4.1",
|
||||||
"core-js": "3.36.0",
|
"core-js": "3.37.0",
|
||||||
"cropperjs": "1.6.1",
|
"cropperjs": "1.6.2",
|
||||||
"date-fns": "2.30.0",
|
"date-fns": "3.6.0",
|
||||||
"date-fns-tz": "2.0.0",
|
"date-fns-tz": "3.1.3",
|
||||||
"deep-clone-simple": "1.1.1",
|
"deep-clone-simple": "1.1.1",
|
||||||
"deep-freeze": "0.0.1",
|
"deep-freeze": "0.0.1",
|
||||||
"element-internals-polyfill": "1.3.10",
|
"element-internals-polyfill": "1.3.11",
|
||||||
"fuse.js": "7.0.0",
|
"fuse.js": "7.0.0",
|
||||||
"google-timezones-json": "1.2.0",
|
"google-timezones-json": "1.2.0",
|
||||||
"hls.js": "1.5.5",
|
"hls.js": "patch:hls.js@npm%3A1.5.7#~/.yarn/patches/hls.js-npm-1.5.7-f5bbd3d060.patch",
|
||||||
"home-assistant-js-websocket": "9.1.0",
|
"home-assistant-js-websocket": "9.3.0",
|
||||||
"idb-keyval": "6.2.1",
|
"idb-keyval": "6.2.1",
|
||||||
"intl-messageformat": "10.5.11",
|
"intl-messageformat": "10.5.11",
|
||||||
"js-yaml": "4.1.0",
|
"js-yaml": "4.1.0",
|
||||||
@@ -118,7 +119,7 @@
|
|||||||
"leaflet-draw": "1.0.4",
|
"leaflet-draw": "1.0.4",
|
||||||
"lit": "2.8.0",
|
"lit": "2.8.0",
|
||||||
"luxon": "3.4.4",
|
"luxon": "3.4.4",
|
||||||
"marked": "12.0.0",
|
"marked": "12.0.2",
|
||||||
"memoize-one": "6.0.0",
|
"memoize-one": "6.0.0",
|
||||||
"node-vibrant": "3.2.1-alpha.1",
|
"node-vibrant": "3.2.1-alpha.1",
|
||||||
"proxy-polyfill": "0.3.2",
|
"proxy-polyfill": "0.3.2",
|
||||||
@@ -129,7 +130,7 @@
|
|||||||
"rrule": "2.8.1",
|
"rrule": "2.8.1",
|
||||||
"sortablejs": "1.15.2",
|
"sortablejs": "1.15.2",
|
||||||
"stacktrace-js": "2.0.2",
|
"stacktrace-js": "2.0.2",
|
||||||
"superstruct": "1.0.3",
|
"superstruct": "1.0.4",
|
||||||
"tinykeys": "2.1.0",
|
"tinykeys": "2.1.0",
|
||||||
"tsparticles-engine": "2.12.0",
|
"tsparticles-engine": "2.12.0",
|
||||||
"tsparticles-preset-links": "2.12.0",
|
"tsparticles-preset-links": "2.12.0",
|
||||||
@@ -146,21 +147,21 @@
|
|||||||
"workbox-precaching": "7.0.0",
|
"workbox-precaching": "7.0.0",
|
||||||
"workbox-routing": "7.0.0",
|
"workbox-routing": "7.0.0",
|
||||||
"workbox-strategies": "7.0.0",
|
"workbox-strategies": "7.0.0",
|
||||||
"xss": "1.0.14"
|
"xss": "1.0.15"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "7.23.9",
|
"@babel/core": "7.24.4",
|
||||||
"@babel/helper-define-polyfill-provider": "0.5.0",
|
"@babel/helper-define-polyfill-provider": "0.6.2",
|
||||||
"@babel/plugin-proposal-decorators": "7.23.9",
|
"@babel/plugin-proposal-decorators": "7.24.1",
|
||||||
"@babel/plugin-transform-runtime": "7.23.9",
|
"@babel/plugin-transform-runtime": "7.24.3",
|
||||||
"@babel/preset-env": "7.23.9",
|
"@babel/preset-env": "7.24.4",
|
||||||
"@babel/preset-typescript": "7.23.3",
|
"@babel/preset-typescript": "7.24.1",
|
||||||
"@bundle-stats/plugin-webpack-filter": "4.10.0",
|
"@bundle-stats/plugin-webpack-filter": "4.12.2",
|
||||||
"@koa/cors": "5.0.0",
|
"@koa/cors": "5.0.0",
|
||||||
"@lokalise/node-api": "12.1.0",
|
"@lokalise/node-api": "12.4.0",
|
||||||
"@octokit/auth-oauth-device": "6.0.1",
|
"@octokit/auth-oauth-device": "7.1.1",
|
||||||
"@octokit/plugin-retry": "6.0.1",
|
"@octokit/plugin-retry": "7.1.0",
|
||||||
"@octokit/rest": "20.0.2",
|
"@octokit/rest": "20.1.0",
|
||||||
"@open-wc/dev-server-hmr": "0.1.4",
|
"@open-wc/dev-server-hmr": "0.1.4",
|
||||||
"@rollup/plugin-babel": "6.0.4",
|
"@rollup/plugin-babel": "6.0.4",
|
||||||
"@rollup/plugin-commonjs": "25.0.7",
|
"@rollup/plugin-commonjs": "25.0.7",
|
||||||
@@ -168,33 +169,33 @@
|
|||||||
"@rollup/plugin-node-resolve": "15.2.3",
|
"@rollup/plugin-node-resolve": "15.2.3",
|
||||||
"@rollup/plugin-replace": "5.0.5",
|
"@rollup/plugin-replace": "5.0.5",
|
||||||
"@types/babel__plugin-transform-runtime": "7.9.5",
|
"@types/babel__plugin-transform-runtime": "7.9.5",
|
||||||
"@types/chromecast-caf-receiver": "6.0.13",
|
"@types/chromecast-caf-receiver": "6.0.14",
|
||||||
"@types/chromecast-caf-sender": "1.0.8",
|
"@types/chromecast-caf-sender": "1.0.9",
|
||||||
|
"@types/color-name": "1.1.4",
|
||||||
"@types/glob": "8.1.0",
|
"@types/glob": "8.1.0",
|
||||||
"@types/gulp-if": "^3",
|
|
||||||
"@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.8",
|
"@types/leaflet": "1.9.12",
|
||||||
"@types/leaflet-draw": "1.0.11",
|
"@types/leaflet-draw": "1.0.11",
|
||||||
"@types/luxon": "3.4.2",
|
"@types/luxon": "3.4.2",
|
||||||
"@types/mocha": "10.0.6",
|
"@types/mocha": "10.0.6",
|
||||||
"@types/qrcode": "1.5.5",
|
"@types/qrcode": "1.5.5",
|
||||||
"@types/serve-handler": "6.1.4",
|
"@types/serve-handler": "6.1.4",
|
||||||
"@types/sortablejs": "1.15.7",
|
"@types/sortablejs": "1.15.8",
|
||||||
"@types/tar": "6.1.11",
|
"@types/tar": "6.1.13",
|
||||||
"@types/ua-parser-js": "0.7.39",
|
"@types/ua-parser-js": "0.7.39",
|
||||||
"@types/webspeechapi": "0.0.29",
|
"@types/webspeechapi": "0.0.29",
|
||||||
"@typescript-eslint/eslint-plugin": "7.0.1",
|
"@typescript-eslint/eslint-plugin": "7.7.1",
|
||||||
"@typescript-eslint/parser": "7.0.1",
|
"@typescript-eslint/parser": "7.7.1",
|
||||||
"@web/dev-server": "0.1.38",
|
"@web/dev-server": "0.1.38",
|
||||||
"@web/dev-server-rollup": "0.4.1",
|
"@web/dev-server-rollup": "0.4.1",
|
||||||
"babel-loader": "9.1.3",
|
"babel-loader": "9.1.3",
|
||||||
"babel-plugin-template-html-minifier": "4.1.0",
|
"babel-plugin-template-html-minifier": "4.1.0",
|
||||||
"chai": "5.1.0",
|
"chai": "5.1.0",
|
||||||
"del": "7.1.0",
|
"del": "7.1.0",
|
||||||
"eslint": "8.56.0",
|
"eslint": "8.57.0",
|
||||||
"eslint-config-airbnb-base": "15.0.0",
|
"eslint-config-airbnb-base": "15.0.0",
|
||||||
"eslint-config-airbnb-typescript": "17.1.0",
|
"eslint-config-airbnb-typescript": "18.0.0",
|
||||||
"eslint-config-prettier": "9.1.0",
|
"eslint-config-prettier": "9.1.0",
|
||||||
"eslint-import-resolver-webpack": "0.13.8",
|
"eslint-import-resolver-webpack": "0.13.8",
|
||||||
"eslint-plugin-disable": "2.0.3",
|
"eslint-plugin-disable": "2.0.3",
|
||||||
@@ -202,15 +203,13 @@
|
|||||||
"eslint-plugin-lit": "1.11.0",
|
"eslint-plugin-lit": "1.11.0",
|
||||||
"eslint-plugin-lit-a11y": "4.1.2",
|
"eslint-plugin-lit-a11y": "4.1.2",
|
||||||
"eslint-plugin-unused-imports": "3.1.0",
|
"eslint-plugin-unused-imports": "3.1.0",
|
||||||
"eslint-plugin-wc": "2.0.4",
|
"eslint-plugin-wc": "2.1.0",
|
||||||
"fancy-log": "2.0.0",
|
"fancy-log": "2.0.0",
|
||||||
"fs-extra": "11.2.0",
|
"fs-extra": "11.2.0",
|
||||||
"glob": "10.3.10",
|
"glob": "10.3.12",
|
||||||
"gulp": "4.0.2",
|
"gulp": "4.0.2",
|
||||||
"gulp-flatmap": "1.0.2",
|
"gulp-json-transform": "0.5.0",
|
||||||
"gulp-if": "3.0.0",
|
"gulp-merge-json": "2.2.1",
|
||||||
"gulp-json-transform": "0.4.8",
|
|
||||||
"gulp-merge-json": "2.1.2",
|
|
||||||
"gulp-rename": "2.0.0",
|
"gulp-rename": "2.0.0",
|
||||||
"gulp-zopfli-green": "6.0.1",
|
"gulp-zopfli-green": "6.0.1",
|
||||||
"html-minifier-terser": "7.2.0",
|
"html-minifier-terser": "7.2.0",
|
||||||
@@ -220,11 +219,11 @@
|
|||||||
"lint-staged": "15.2.2",
|
"lint-staged": "15.2.2",
|
||||||
"lit-analyzer": "2.0.3",
|
"lit-analyzer": "2.0.3",
|
||||||
"lodash.template": "4.5.0",
|
"lodash.template": "4.5.0",
|
||||||
"magic-string": "0.30.7",
|
"magic-string": "0.30.10",
|
||||||
"map-stream": "0.0.7",
|
"map-stream": "0.0.7",
|
||||||
"mocha": "10.3.0",
|
"mocha": "10.4.0",
|
||||||
"object-hash": "3.0.0",
|
"object-hash": "3.0.0",
|
||||||
"open": "10.0.3",
|
"open": "10.1.0",
|
||||||
"pinst": "3.0.0",
|
"pinst": "3.0.0",
|
||||||
"prettier": "3.2.5",
|
"prettier": "3.2.5",
|
||||||
"rollup": "2.79.1",
|
"rollup": "2.79.1",
|
||||||
@@ -235,20 +234,17 @@
|
|||||||
"sinon": "17.0.1",
|
"sinon": "17.0.1",
|
||||||
"source-map-url": "0.4.1",
|
"source-map-url": "0.4.1",
|
||||||
"systemjs": "6.14.3",
|
"systemjs": "6.14.3",
|
||||||
"tar": "6.2.0",
|
"tar": "7.0.1",
|
||||||
"terser-webpack-plugin": "5.3.10",
|
"terser-webpack-plugin": "5.3.10",
|
||||||
"transform-async-modules-webpack-plugin": "1.0.2",
|
"transform-async-modules-webpack-plugin": "1.1.0",
|
||||||
"ts-lit-plugin": "2.0.2",
|
"ts-lit-plugin": "2.0.2",
|
||||||
"typescript": "5.3.3",
|
"typescript": "5.4.5",
|
||||||
"vinyl-buffer": "1.0.1",
|
"webpack": "5.91.0",
|
||||||
"vinyl-paths": "5.0.0",
|
|
||||||
"vinyl-source-stream": "2.0.0",
|
|
||||||
"webpack": "5.90.2",
|
|
||||||
"webpack-cli": "5.1.4",
|
"webpack-cli": "5.1.4",
|
||||||
"webpack-dev-server": "4.15.1",
|
"webpack-dev-server": "5.0.4",
|
||||||
"webpack-manifest-plugin": "5.0.0",
|
"webpack-manifest-plugin": "5.0.0",
|
||||||
"webpack-stats-plugin": "1.1.3",
|
"webpack-stats-plugin": "1.1.3",
|
||||||
"webpackbar": "6.0.0",
|
"webpackbar": "6.0.1",
|
||||||
"workbox-build": "7.0.0"
|
"workbox-build": "7.0.0"
|
||||||
},
|
},
|
||||||
"_comment": "Polymer 3.2 contained a bug, fixed in https://github.com/Polymer/polymer/pull/5569, add as patch",
|
"_comment": "Polymer 3.2 contained a bug, fixed in https://github.com/Polymer/polymer/pull/5569, add as patch",
|
||||||
@@ -261,5 +257,5 @@
|
|||||||
"sortablejs@1.15.2": "patch:sortablejs@npm%3A1.15.2#~/.yarn/patches/sortablejs-npm-1.15.2-73347ae85a.patch",
|
"sortablejs@1.15.2": "patch:sortablejs@npm%3A1.15.2#~/.yarn/patches/sortablejs-npm-1.15.2-73347ae85a.patch",
|
||||||
"leaflet-draw@1.0.4": "patch:leaflet-draw@npm%3A1.0.4#./.yarn/patches/leaflet-draw-npm-1.0.4-0ca0ebcf65.patch"
|
"leaflet-draw@1.0.4": "patch:leaflet-draw@npm%3A1.0.4#./.yarn/patches/leaflet-draw-npm-1.0.4-0ca0ebcf65.patch"
|
||||||
},
|
},
|
||||||
"packageManager": "yarn@4.1.0"
|
"packageManager": "yarn@4.1.1"
|
||||||
}
|
}
|
||||||
|
1
public/static/images/appstore.svg
Normal file
1
public/static/images/appstore.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 7.1 KiB |
BIN
public/static/images/logo_apple_home.png
Normal file
BIN
public/static/images/logo_apple_home.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 19 KiB |
BIN
public/static/images/logo_google_home.png
Normal file
BIN
public/static/images/logo_google_home.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 8.6 KiB |
1
public/static/images/playstore.svg
Normal file
1
public/static/images/playstore.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 6.2 KiB |
1
public/static/images/qr-appstore.svg
Normal file
1
public/static/images/qr-appstore.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 52 KiB |
1
public/static/images/qr-playstore.svg
Normal file
1
public/static/images/qr-playstore.svg
Normal file
File diff suppressed because one or more lines are too long
After Width: | Height: | Size: 69 KiB |
@@ -4,14 +4,14 @@ build-backend = "setuptools.build_meta"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "home-assistant-frontend"
|
name = "home-assistant-frontend"
|
||||||
version = "20240207.0"
|
version = "20240426.0"
|
||||||
license = {text = "Apache-2.0"}
|
license = {text = "Apache-2.0"}
|
||||||
description = "The Home Assistant frontend"
|
description = "The Home Assistant frontend"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
authors = [
|
authors = [
|
||||||
{name = "The Home Assistant Authors", email = "hello@home-assistant.io"}
|
{name = "The Home Assistant Authors", email = "hello@home-assistant.io"}
|
||||||
]
|
]
|
||||||
requires-python = ">=3.10.0"
|
requires-python = ">=3.11.0"
|
||||||
|
|
||||||
[project.urls]
|
[project.urls]
|
||||||
"Homepage" = "https://github.com/home-assistant/frontend"
|
"Homepage" = "https://github.com/home-assistant/frontend"
|
||||||
|
@@ -40,6 +40,11 @@
|
|||||||
"matchPackageNames": ["tsparticles-engine"],
|
"matchPackageNames": ["tsparticles-engine"],
|
||||||
"matchPackagePrefixes": ["tsparticles-preset-"]
|
"matchPackagePrefixes": ["tsparticles-preset-"]
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"description": "Group date-fns with dependent timezone package",
|
||||||
|
"groupName": "date-fns",
|
||||||
|
"matchPackageNames": ["date-fns", "date-fns-tz"]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"description": "Group and temporarily disable WDS packages",
|
"description": "Group and temporarily disable WDS packages",
|
||||||
"groupName": "Web Dev Server",
|
"groupName": "Web Dev Server",
|
||||||
|
@@ -1,3 +1,5 @@
|
|||||||
|
import { theme2hex } from "./convert-color";
|
||||||
|
|
||||||
export const COLORS = [
|
export const COLORS = [
|
||||||
"#44739e",
|
"#44739e",
|
||||||
"#984ea3",
|
"#984ea3",
|
||||||
@@ -65,10 +67,10 @@ export function getColorByIndex(index: number) {
|
|||||||
export function getGraphColorByIndex(
|
export function getGraphColorByIndex(
|
||||||
index: number,
|
index: number,
|
||||||
style: CSSStyleDeclaration
|
style: CSSStyleDeclaration
|
||||||
) {
|
): string {
|
||||||
// The CSS vars for the colors use range 1..n, so we need to adjust the index from the internal 0..n color index range.
|
// The CSS vars for the colors use range 1..n, so we need to adjust the index from the internal 0..n color index range.
|
||||||
return (
|
const themeColor =
|
||||||
style.getPropertyValue(`--graph-color-${index + 1}`) ||
|
style.getPropertyValue(`--graph-color-${index + 1}`) ||
|
||||||
getColorByIndex(index)
|
getColorByIndex(index);
|
||||||
);
|
return theme2hex(themeColor);
|
||||||
}
|
}
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
import colors from "color-name";
|
||||||
import { expandHex } from "./hex";
|
import { expandHex } from "./hex";
|
||||||
|
|
||||||
const rgb_hex = (component: number): string => {
|
const rgb_hex = (component: number): string => {
|
||||||
@@ -126,3 +127,18 @@ export const rgb2hs = (rgb: [number, number, number]): [number, number] =>
|
|||||||
|
|
||||||
export const hs2rgb = (hs: [number, number]): [number, number, number] =>
|
export const hs2rgb = (hs: [number, number]): [number, number, number] =>
|
||||||
hsv2rgb([hs[0], hs[1], 255]);
|
hsv2rgb([hs[0], hs[1], 255]);
|
||||||
|
|
||||||
|
export function theme2hex(themeColor: string): string {
|
||||||
|
if (themeColor.startsWith("#")) {
|
||||||
|
return themeColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
const rgbFromColorName = colors[themeColor];
|
||||||
|
if (!rgbFromColorName) {
|
||||||
|
// We have a named color, and there's nothing in the table,
|
||||||
|
// so nothing further we can do with it.
|
||||||
|
// Compare/border/background color will all be the same.
|
||||||
|
return themeColor;
|
||||||
|
}
|
||||||
|
return rgb2hex(rgbFromColorName);
|
||||||
|
}
|
||||||
|
@@ -1,19 +1,25 @@
|
|||||||
import { PageNavigation } from "../../layouts/hass-tabs-subpage";
|
import { PageNavigation } from "../../layouts/hass-tabs-subpage";
|
||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
|
import { ensureArray } from "../array/ensure-array";
|
||||||
import { isComponentLoaded } from "./is_component_loaded";
|
import { isComponentLoaded } from "./is_component_loaded";
|
||||||
|
|
||||||
export const canShowPage = (hass: HomeAssistant, page: PageNavigation) =>
|
export const canShowPage = (hass: HomeAssistant, page: PageNavigation) =>
|
||||||
(isCore(page) || isLoadedIntegration(hass, page)) &&
|
(isCore(page) || isLoadedIntegration(hass, page)) &&
|
||||||
!hideAdvancedPage(hass, page);
|
!hideAdvancedPage(hass, page) &&
|
||||||
|
isNotLoadedIntegration(hass, page);
|
||||||
|
|
||||||
const isLoadedIntegration = (hass: HomeAssistant, page: PageNavigation) =>
|
const isLoadedIntegration = (hass: HomeAssistant, page: PageNavigation) =>
|
||||||
page.component
|
!page.component ||
|
||||||
? isComponentLoaded(hass, page.component)
|
ensureArray(page.component).some((integration) =>
|
||||||
: page.components
|
isComponentLoaded(hass, integration)
|
||||||
? page.components.some((integration) =>
|
);
|
||||||
isComponentLoaded(hass, integration)
|
|
||||||
)
|
const isNotLoadedIntegration = (hass: HomeAssistant, page: PageNavigation) =>
|
||||||
: true;
|
!page.not_component ||
|
||||||
|
!ensureArray(page.not_component).some((integration) =>
|
||||||
|
isComponentLoaded(hass, integration)
|
||||||
|
);
|
||||||
|
|
||||||
const isCore = (page: PageNavigation) => page.core;
|
const isCore = (page: PageNavigation) => page.core;
|
||||||
const isAdvancedPage = (page: PageNavigation) => page.advancedOnly;
|
const isAdvancedPage = (page: PageNavigation) => page.advancedOnly;
|
||||||
const userWantsAdvanced = (hass: HomeAssistant) => hass.userData?.showAdvanced;
|
const userWantsAdvanced = (hass: HomeAssistant) => hass.userData?.showAdvanced;
|
||||||
|
@@ -231,6 +231,7 @@ export const SENSOR_ENTITIES = [
|
|||||||
"calendar",
|
"calendar",
|
||||||
"camera",
|
"camera",
|
||||||
"device_tracker",
|
"device_tracker",
|
||||||
|
"image",
|
||||||
"weather",
|
"weather",
|
||||||
];
|
];
|
||||||
|
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { utcToZonedTime, zonedTimeToUtc } from "date-fns-tz";
|
import { toZonedTime, fromZonedTime } from "date-fns-tz";
|
||||||
import { HassConfig } from "home-assistant-js-websocket";
|
import { HassConfig } from "home-assistant-js-websocket";
|
||||||
import { FrontendLocaleData, TimeZone } from "../../data/translation";
|
import { FrontendLocaleData, TimeZone } from "../../data/translation";
|
||||||
|
|
||||||
@@ -8,10 +8,10 @@ const calcZonedDate = (
|
|||||||
fn: (date: Date, options?: any) => Date | number | boolean,
|
fn: (date: Date, options?: any) => Date | number | boolean,
|
||||||
options?
|
options?
|
||||||
) => {
|
) => {
|
||||||
const inputZoned = utcToZonedTime(date, tz);
|
const inputZoned = toZonedTime(date, tz);
|
||||||
const fnZoned = fn(inputZoned, options);
|
const fnZoned = fn(inputZoned, options);
|
||||||
if (fnZoned instanceof Date) {
|
if (fnZoned instanceof Date) {
|
||||||
return zonedTimeToUtc(fnZoned, tz) as Date;
|
return fromZonedTime(fnZoned, tz) as Date;
|
||||||
}
|
}
|
||||||
return fnZoned;
|
return fnZoned;
|
||||||
};
|
};
|
||||||
@@ -37,3 +37,20 @@ export const calcDateProperty = (
|
|||||||
locale.time_zone === TimeZone.server
|
locale.time_zone === TimeZone.server
|
||||||
? (calcZonedDate(date, config.time_zone, fn, options) as number | boolean)
|
? (calcZonedDate(date, config.time_zone, fn, options) as number | boolean)
|
||||||
: fn(date, options);
|
: fn(date, options);
|
||||||
|
|
||||||
|
export const calcDateDifferenceProperty = (
|
||||||
|
endDate: Date,
|
||||||
|
startDate: Date,
|
||||||
|
fn: (date: Date, options?: any) => boolean | number,
|
||||||
|
locale: FrontendLocaleData,
|
||||||
|
config: HassConfig
|
||||||
|
) =>
|
||||||
|
calcDateProperty(
|
||||||
|
endDate,
|
||||||
|
fn,
|
||||||
|
locale,
|
||||||
|
config,
|
||||||
|
locale.time_zone === TimeZone.server
|
||||||
|
? toZonedTime(startDate, config.time_zone)
|
||||||
|
: startDate
|
||||||
|
);
|
||||||
|
@@ -1,8 +1,13 @@
|
|||||||
import { MAIN_WINDOW_NAME } from "../../data/main_window";
|
import { MAIN_WINDOW_NAME } from "../../data/main_window";
|
||||||
|
|
||||||
export const mainWindow =
|
export const mainWindow = (() => {
|
||||||
window.name === MAIN_WINDOW_NAME
|
try {
|
||||||
? window
|
return window.name === MAIN_WINDOW_NAME
|
||||||
: parent.name === MAIN_WINDOW_NAME
|
? window
|
||||||
? parent
|
: parent.name === MAIN_WINDOW_NAME
|
||||||
: top!;
|
? parent
|
||||||
|
: top!;
|
||||||
|
} catch {
|
||||||
|
return window;
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
@@ -187,11 +187,14 @@ export const computeStateDisplayFromEntityAttributes = (
|
|||||||
if (
|
if (
|
||||||
[
|
[
|
||||||
"button",
|
"button",
|
||||||
|
"conversation",
|
||||||
"event",
|
"event",
|
||||||
"image",
|
"image",
|
||||||
"input_button",
|
"input_button",
|
||||||
|
"notify",
|
||||||
"scene",
|
"scene",
|
||||||
"stt",
|
"stt",
|
||||||
|
"tag",
|
||||||
"tts",
|
"tts",
|
||||||
"wake_word",
|
"wake_word",
|
||||||
].includes(domain) ||
|
].includes(domain) ||
|
||||||
|
@@ -2,6 +2,7 @@ import IntlMessageFormat from "intl-messageformat";
|
|||||||
import type { HTMLTemplateResult } from "lit";
|
import type { HTMLTemplateResult } from "lit";
|
||||||
import { polyfillLocaleData } from "../../resources/locale-data-polyfill";
|
import { polyfillLocaleData } from "../../resources/locale-data-polyfill";
|
||||||
import { Resources, TranslationDict } from "../../types";
|
import { Resources, TranslationDict } from "../../types";
|
||||||
|
import { fireEvent } from "../dom/fire_event";
|
||||||
|
|
||||||
// Exclude some patterns from key type checking for now
|
// Exclude some patterns from key type checking for now
|
||||||
// These are intended to be removed as errors are fixed
|
// These are intended to be removed as errors are fixed
|
||||||
@@ -81,7 +82,9 @@ export interface FormatsType {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
export const computeLocalize = async <Keys extends string = LocalizeKeys>(
|
export const computeLocalize = async <Keys extends string = LocalizeKeys>(
|
||||||
cache: any,
|
cache: HTMLElement & {
|
||||||
|
_localizationCache?: Record<string, IntlMessageFormat>;
|
||||||
|
},
|
||||||
language: string,
|
language: string,
|
||||||
resources: Resources,
|
resources: Resources,
|
||||||
formats?: FormatsType
|
formats?: FormatsType
|
||||||
@@ -107,7 +110,7 @@ export const computeLocalize = async <Keys extends string = LocalizeKeys>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
const messageKey = key + translatedValue;
|
const messageKey = key + translatedValue;
|
||||||
let translatedMessage = cache._localizationCache[messageKey] as
|
let translatedMessage = cache._localizationCache![messageKey] as
|
||||||
| IntlMessageFormat
|
| IntlMessageFormat
|
||||||
| undefined;
|
| undefined;
|
||||||
|
|
||||||
@@ -121,7 +124,7 @@ export const computeLocalize = async <Keys extends string = LocalizeKeys>(
|
|||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
return "Translation error: " + err.message;
|
return "Translation error: " + err.message;
|
||||||
}
|
}
|
||||||
cache._localizationCache[messageKey] = translatedMessage;
|
cache._localizationCache![messageKey] = translatedMessage;
|
||||||
}
|
}
|
||||||
|
|
||||||
let argObject = {};
|
let argObject = {};
|
||||||
@@ -137,6 +140,12 @@ export const computeLocalize = async <Keys extends string = LocalizeKeys>(
|
|||||||
try {
|
try {
|
||||||
return translatedMessage.format<string>(argObject) as string;
|
return translatedMessage.format<string>(argObject) as string;
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.error("Translation error", key, language, err);
|
||||||
|
fireEvent(cache, "write_log", {
|
||||||
|
level: "error",
|
||||||
|
message: `Failed to format translation for key '${key}' in language '${language}'. ${err}`,
|
||||||
|
});
|
||||||
return "Translation " + err;
|
return "Translation " + err;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@@ -20,14 +20,14 @@ function findNestedItem(
|
|||||||
}, obj);
|
}, obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function nestedArrayMove<T>(
|
export function nestedArrayMove<A>(
|
||||||
obj: T | T[],
|
obj: A,
|
||||||
oldIndex: number,
|
oldIndex: number,
|
||||||
newIndex: number,
|
newIndex: number,
|
||||||
oldPath?: ItemPath,
|
oldPath?: ItemPath,
|
||||||
newPath?: ItemPath
|
newPath?: ItemPath
|
||||||
): T | T[] {
|
): A {
|
||||||
const newObj = Array.isArray(obj) ? [...obj] : { ...obj };
|
const newObj = (Array.isArray(obj) ? [...obj] : { ...obj }) as A;
|
||||||
const from = oldPath ? findNestedItem(newObj, oldPath) : newObj;
|
const from = oldPath ? findNestedItem(newObj, oldPath) : newObj;
|
||||||
const to = newPath ? findNestedItem(newObj, newPath, true) : newObj;
|
const to = newPath ? findNestedItem(newObj, newPath, true) : newObj;
|
||||||
|
|
||||||
|
9
src/common/util/promise-all-settled-results.ts
Normal file
9
src/common/util/promise-all-settled-results.ts
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
export const hasRejectedItems = <T = any>(results: PromiseSettledResult<T>[]) =>
|
||||||
|
results.some((result) => result.status === "rejected");
|
||||||
|
|
||||||
|
export const rejectedItems = <T = any>(
|
||||||
|
results: PromiseSettledResult<T>[]
|
||||||
|
): PromiseRejectedResult[] =>
|
||||||
|
results.filter(
|
||||||
|
(result) => result.status === "rejected"
|
||||||
|
) as PromiseRejectedResult[];
|
@@ -1,4 +1,4 @@
|
|||||||
import { differenceInDays, differenceInWeeks, startOfWeek } from "date-fns/esm";
|
import { differenceInDays, differenceInWeeks, startOfWeek } from "date-fns";
|
||||||
import { FrontendLocaleData } from "../../data/translation";
|
import { FrontendLocaleData } from "../../data/translation";
|
||||||
import { firstWeekdayIndex } from "../datetime/first_weekday";
|
import { firstWeekdayIndex } from "../datetime/first_weekday";
|
||||||
|
|
||||||
|
@@ -34,7 +34,7 @@ import {
|
|||||||
endOfMonth,
|
endOfMonth,
|
||||||
endOfQuarter,
|
endOfQuarter,
|
||||||
endOfYear,
|
endOfYear,
|
||||||
} from "date-fns/esm";
|
} from "date-fns";
|
||||||
import {
|
import {
|
||||||
formatDate,
|
formatDate,
|
||||||
formatDateMonth,
|
formatDateMonth,
|
||||||
|
@@ -75,6 +75,8 @@ export class HaChartBase extends LitElement {
|
|||||||
|
|
||||||
private _paddingYAxisInternal = 0;
|
private _paddingYAxisInternal = 0;
|
||||||
|
|
||||||
|
private _datasetOrder: number[] = [];
|
||||||
|
|
||||||
public disconnectedCallback() {
|
public disconnectedCallback() {
|
||||||
super.disconnectedCallback();
|
super.disconnectedCallback();
|
||||||
this._releaseCanvas();
|
this._releaseCanvas();
|
||||||
@@ -165,7 +167,17 @@ export class HaChartBase extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// put the legend labels in sorted order if provided
|
||||||
if (changedProps.has("data")) {
|
if (changedProps.has("data")) {
|
||||||
|
this._datasetOrder = this.data.datasets.map((_, index) => index);
|
||||||
|
if (this.data?.datasets.some((dataset) => dataset.order)) {
|
||||||
|
this._datasetOrder.sort(
|
||||||
|
(a, b) =>
|
||||||
|
(this.data.datasets[a].order || 0) -
|
||||||
|
(this.data.datasets[b].order || 0)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (this.externalHidden) {
|
if (this.externalHidden) {
|
||||||
this._hiddenDatasets = new Set();
|
this._hiddenDatasets = new Set();
|
||||||
if (this.data?.datasets) {
|
if (this.data?.datasets) {
|
||||||
@@ -205,8 +217,9 @@ export class HaChartBase extends LitElement {
|
|||||||
${this.options?.plugins?.legend?.display === true
|
${this.options?.plugins?.legend?.display === true
|
||||||
? html`<div class="chartLegend">
|
? html`<div class="chartLegend">
|
||||||
<ul>
|
<ul>
|
||||||
${this.data.datasets.map((dataset, index) =>
|
${this._datasetOrder.map((index) => {
|
||||||
this.extraData?.[index]?.show_legend === false
|
const dataset = this.data.datasets[index];
|
||||||
|
return this.extraData?.[index]?.show_legend === false
|
||||||
? nothing
|
? nothing
|
||||||
: html`<li
|
: html`<li
|
||||||
.datasetIndex=${index}
|
.datasetIndex=${index}
|
||||||
@@ -228,8 +241,8 @@ export class HaChartBase extends LitElement {
|
|||||||
${this.extraData?.[index]?.legend_label ??
|
${this.extraData?.[index]?.legend_label ??
|
||||||
dataset.label}
|
dataset.label}
|
||||||
</div>
|
</div>
|
||||||
</li>`
|
</li>`;
|
||||||
)}
|
})}
|
||||||
</ul>
|
</ul>
|
||||||
</div>`
|
</div>`
|
||||||
: ""}
|
: ""}
|
||||||
|
@@ -111,7 +111,7 @@ export class StateHistoryChartLine extends LitElement {
|
|||||||
config: this.hass.config,
|
config: this.hass.config,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
suggestedMin: this.startTime,
|
min: this.startTime,
|
||||||
suggestedMax: this.endTime,
|
suggestedMax: this.endTime,
|
||||||
ticks: {
|
ticks: {
|
||||||
maxRotation: 0,
|
maxRotation: 0,
|
||||||
|
@@ -114,7 +114,7 @@ export class StateHistoryChartTimeline extends LitElement {
|
|||||||
config: this.hass.config,
|
config: this.hass.config,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
suggestedMin: this.startTime,
|
min: this.startTime,
|
||||||
suggestedMax: this.endTime,
|
suggestedMax: this.endTime,
|
||||||
ticks: {
|
ticks: {
|
||||||
autoSkip: true,
|
autoSkip: true,
|
||||||
|
@@ -233,16 +233,32 @@ export class StateHistoryCharts extends LitElement {
|
|||||||
new Date().getTime() - 60 * 60 * this.hoursToShow * 1000
|
new Date().getTime() - 60 * 60 * this.hoursToShow * 1000
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
this._computedStartTime = new Date(
|
let minTimeAll = (this.historyData?.timeline ?? []).reduce(
|
||||||
(this.historyData?.timeline ?? []).reduce(
|
(minTime, stateInfo) =>
|
||||||
(minTime, stateInfo) =>
|
Math.min(
|
||||||
Math.min(
|
minTime,
|
||||||
minTime,
|
new Date(stateInfo.data[0].last_changed).getTime()
|
||||||
new Date(stateInfo.data[0].last_changed).getTime()
|
),
|
||||||
),
|
new Date().getTime()
|
||||||
new Date().getTime()
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
minTimeAll = (this.historyData?.line ?? []).reduce(
|
||||||
|
(minTimeLine, line) =>
|
||||||
|
Math.min(
|
||||||
|
minTimeLine,
|
||||||
|
line.data.reduce(
|
||||||
|
(minTimeData, data) =>
|
||||||
|
Math.min(
|
||||||
|
minTimeData,
|
||||||
|
new Date(data.states[0].last_changed).getTime()
|
||||||
|
),
|
||||||
|
minTimeLine
|
||||||
|
)
|
||||||
|
),
|
||||||
|
minTimeAll
|
||||||
|
);
|
||||||
|
|
||||||
|
this._computedStartTime = new Date(minTimeAll);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -16,6 +16,7 @@ import { customElement, property, state, query } from "lit/decorators";
|
|||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { getGraphColorByIndex } from "../../common/color/colors";
|
import { getGraphColorByIndex } from "../../common/color/colors";
|
||||||
import { isComponentLoaded } from "../../common/config/is_component_loaded";
|
import { isComponentLoaded } from "../../common/config/is_component_loaded";
|
||||||
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
import {
|
import {
|
||||||
formatNumber,
|
formatNumber,
|
||||||
numberFormatToLocale,
|
numberFormatToLocale,
|
||||||
@@ -25,6 +26,7 @@ import {
|
|||||||
getDisplayUnit,
|
getDisplayUnit,
|
||||||
getStatisticLabel,
|
getStatisticLabel,
|
||||||
getStatisticMetadata,
|
getStatisticMetadata,
|
||||||
|
isExternalStatistic,
|
||||||
Statistics,
|
Statistics,
|
||||||
statisticsHaveType,
|
statisticsHaveType,
|
||||||
StatisticsMetaData,
|
StatisticsMetaData,
|
||||||
@@ -79,6 +81,8 @@ export class StatisticsChart extends LitElement {
|
|||||||
|
|
||||||
@property({ type: Boolean }) public isLoadingData = false;
|
@property({ type: Boolean }) public isLoadingData = false;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public clickForMoreInfo = true;
|
||||||
|
|
||||||
@property() public period?: string;
|
@property() public period?: string;
|
||||||
|
|
||||||
@state() private _chartData: ChartData = { datasets: [] };
|
@state() private _chartData: ChartData = { datasets: [] };
|
||||||
@@ -273,6 +277,33 @@ export class StatisticsChart extends LitElement {
|
|||||||
},
|
},
|
||||||
// @ts-expect-error
|
// @ts-expect-error
|
||||||
locale: numberFormatToLocale(this.hass.locale),
|
locale: numberFormatToLocale(this.hass.locale),
|
||||||
|
onClick: (e: any) => {
|
||||||
|
if (
|
||||||
|
!this.clickForMoreInfo ||
|
||||||
|
!(e.native instanceof MouseEvent) ||
|
||||||
|
(e.native instanceof PointerEvent && e.native.pointerType !== "mouse")
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const chart = e.chart;
|
||||||
|
|
||||||
|
const points = chart.getElementsAtEventForMode(
|
||||||
|
e,
|
||||||
|
"nearest",
|
||||||
|
{ intersect: true },
|
||||||
|
true
|
||||||
|
);
|
||||||
|
|
||||||
|
if (points.length) {
|
||||||
|
const firstPoint = points[0];
|
||||||
|
const statisticId = this._statisticIds[firstPoint.datasetIndex];
|
||||||
|
if (!isExternalStatistic(statisticId)) {
|
||||||
|
fireEvent(this, "hass-more-info", { entityId: statisticId });
|
||||||
|
chart.canvas.dispatchEvent(new Event("mouseout")); // to hide tooltip
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -205,7 +205,9 @@ export class TimelineController extends BarController {
|
|||||||
|
|
||||||
const y = vScale.getPixelForValue(this.index);
|
const y = vScale.getPixelForValue(this.index);
|
||||||
|
|
||||||
const xStart = iScale.getPixelForValue(data.start.getTime());
|
const xStart = iScale.getPixelForValue(
|
||||||
|
Math.max(iScale.min, data.start.getTime())
|
||||||
|
);
|
||||||
const xEnd = iScale.getPixelForValue(data.end.getTime());
|
const xEnd = iScale.getPixelForValue(data.end.getTime());
|
||||||
const width = xEnd - xStart;
|
const width = xEnd - xStart;
|
||||||
|
|
||||||
|
@@ -49,7 +49,7 @@ export class TimeLineScale extends TimeScale {
|
|||||||
max = isFinite(max) && !isNaN(max) ? max : +adapter.endOf(Date.now(), unit);
|
max = isFinite(max) && !isNaN(max) ? max : +adapter.endOf(Date.now(), unit);
|
||||||
|
|
||||||
// Make sure that max is strictly higher than min (required by the lookup table)
|
// Make sure that max is strictly higher than min (required by the lookup table)
|
||||||
this.min = Math.min(min, max - 1);
|
this.min = adapter.parse(options.min, this) ?? Math.min(min, max - 1);
|
||||||
this.max = Math.max(min + 1, max);
|
this.max = adapter.parse(options.max, this) ?? Math.max(min + 1, max);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -4,22 +4,24 @@ import { css, html } from "lit";
|
|||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
|
|
||||||
@customElement("ha-assist-chip")
|
@customElement("ha-assist-chip")
|
||||||
|
// @ts-ignore
|
||||||
export class HaAssistChip extends MdAssistChip {
|
export class HaAssistChip extends MdAssistChip {
|
||||||
@property({ type: Boolean, reflect: true }) filled = false;
|
@property({ type: Boolean, reflect: true }) filled = false;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) active = false;
|
||||||
|
|
||||||
static override styles = [
|
static override styles = [
|
||||||
...super.styles,
|
...super.styles,
|
||||||
css`
|
css`
|
||||||
:host {
|
:host {
|
||||||
--md-sys-color-primary: var(--primary-text-color);
|
--md-sys-color-primary: var(--primary-text-color);
|
||||||
--md-sys-color-on-surface: var(--primary-text-color);
|
--md-sys-color-on-surface: var(--primary-text-color);
|
||||||
--md-assist-chip-container-shape: 16px;
|
--md-assist-chip-container-shape: var(
|
||||||
|
--ha-assist-chip-container-shape,
|
||||||
|
16px
|
||||||
|
);
|
||||||
--md-assist-chip-outline-color: var(--outline-color);
|
--md-assist-chip-outline-color: var(--outline-color);
|
||||||
--md-assist-chip-label-text-weight: 400;
|
--md-assist-chip-label-text-weight: 400;
|
||||||
--ha-assist-chip-filled-container-color: rgba(
|
|
||||||
var(--rgb-primary-text-color),
|
|
||||||
0.15
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
/** Material 3 doesn't have a filled chip, so we have to make our own **/
|
/** Material 3 doesn't have a filled chip, so we have to make our own **/
|
||||||
.filled {
|
.filled {
|
||||||
@@ -31,10 +33,28 @@ export class HaAssistChip extends MdAssistChip {
|
|||||||
background-color: var(--ha-assist-chip-filled-container-color);
|
background-color: var(--ha-assist-chip-filled-container-color);
|
||||||
}
|
}
|
||||||
/** Set the size of mdc icons **/
|
/** Set the size of mdc icons **/
|
||||||
::slotted([slot="icon"]) {
|
::slotted([slot="icon"]),
|
||||||
|
::slotted([slot="trailingIcon"]) {
|
||||||
display: flex;
|
display: flex;
|
||||||
--mdc-icon-size: var(--md-input-chip-icon-size, 18px);
|
--mdc-icon-size: var(--md-input-chip-icon-size, 18px);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.trailing.icon ::slotted(*),
|
||||||
|
.trailing.icon svg {
|
||||||
|
margin-inline-end: unset;
|
||||||
|
margin-inline-start: var(--_icon-label-space);
|
||||||
|
}
|
||||||
|
::before {
|
||||||
|
background: var(--ha-assist-chip-container-color, transparent);
|
||||||
|
opacity: var(--ha-assist-chip-container-opacity, 1);
|
||||||
|
}
|
||||||
|
:where(.active)::before {
|
||||||
|
background: var(--ha-assist-chip-active-container-color);
|
||||||
|
opacity: var(--ha-assist-chip-active-container-opacity);
|
||||||
|
}
|
||||||
|
.label {
|
||||||
|
font-family: Roboto, sans-serif;
|
||||||
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -45,6 +65,30 @@ export class HaAssistChip extends MdAssistChip {
|
|||||||
|
|
||||||
return super.renderOutline();
|
return super.renderOutline();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected override getContainerClasses() {
|
||||||
|
return {
|
||||||
|
...super.getContainerClasses(),
|
||||||
|
active: this.active,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override renderPrimaryContent() {
|
||||||
|
return html`
|
||||||
|
<span class="leading icon" aria-hidden="true">
|
||||||
|
${this.renderLeadingIcon()}
|
||||||
|
</span>
|
||||||
|
<span class="label">${this.label}</span>
|
||||||
|
<span class="touch"></span>
|
||||||
|
<span class="trailing leading icon" aria-hidden="true">
|
||||||
|
${this.renderTrailingIcon()}
|
||||||
|
</span>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected renderTrailingIcon() {
|
||||||
|
return html`<slot name="trailing-icon"></slot>`;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
|
@@ -19,12 +19,16 @@ export class HaInputChip extends MdInputChip {
|
|||||||
var(--rgb-primary-text-color),
|
var(--rgb-primary-text-color),
|
||||||
0.15
|
0.15
|
||||||
);
|
);
|
||||||
|
--ha-input-chip-selected-container-opacity: 1;
|
||||||
}
|
}
|
||||||
/** Set the size of mdc icons **/
|
/** Set the size of mdc icons **/
|
||||||
::slotted([slot="icon"]) {
|
::slotted([slot="icon"]) {
|
||||||
display: flex;
|
display: flex;
|
||||||
--mdc-icon-size: var(--md-input-chip-icon-size, 18px);
|
--mdc-icon-size: var(--md-input-chip-icon-size, 18px);
|
||||||
}
|
}
|
||||||
|
.selected::before {
|
||||||
|
opacity: var(--ha-input-chip-selected-container-opacity);
|
||||||
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
@@ -44,6 +44,8 @@ class HaDataTableIcon extends LitElement {
|
|||||||
div {
|
div {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
right: 28px;
|
right: 28px;
|
||||||
|
inset-inline-end: 28px;
|
||||||
|
inset-inline-start: initial;
|
||||||
z-index: 1002;
|
z-index: 1002;
|
||||||
outline: none;
|
outline: none;
|
||||||
font-size: 10px;
|
font-size: 10px;
|
||||||
|
130
src/components/data-table/ha-data-table-labels.ts
Normal file
130
src/components/data-table/ha-data-table-labels.ts
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
import { css, html, LitElement, nothing, TemplateResult } from "lit";
|
||||||
|
import { customElement, property } from "lit/decorators";
|
||||||
|
import { repeat } from "lit/directives/repeat";
|
||||||
|
import { LabelRegistryEntry } from "../../data/label_registry";
|
||||||
|
import { computeCssColor } from "../../common/color/compute-color";
|
||||||
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
|
import "../ha-label";
|
||||||
|
import { stringCompare } from "../../common/string/compare";
|
||||||
|
|
||||||
|
@customElement("ha-data-table-labels")
|
||||||
|
class HaDataTableLabels extends LitElement {
|
||||||
|
@property({ attribute: false }) public labels!: LabelRegistryEntry[];
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
const labels = this.labels.sort((a, b) => stringCompare(a.name, b.name));
|
||||||
|
return html`
|
||||||
|
<ha-chip-set>
|
||||||
|
${repeat(
|
||||||
|
labels.slice(0, 2),
|
||||||
|
(label) => label.label_id,
|
||||||
|
(label) => this._renderLabel(label, true)
|
||||||
|
)}
|
||||||
|
${labels.length > 2
|
||||||
|
? html`<ha-button-menu
|
||||||
|
absolute
|
||||||
|
role="button"
|
||||||
|
tabindex="0"
|
||||||
|
@click=${this._handleIconOverflowMenuOpened}
|
||||||
|
@closed=${this._handleIconOverflowMenuClosed}
|
||||||
|
>
|
||||||
|
<ha-label slot="trigger" class="plus" dense>
|
||||||
|
+${labels.length - 2}
|
||||||
|
</ha-label>
|
||||||
|
${repeat(
|
||||||
|
labels.slice(2),
|
||||||
|
(label) => label.label_id,
|
||||||
|
(label) => html`
|
||||||
|
<ha-list-item @click=${this._labelClicked} .item=${label}>
|
||||||
|
${this._renderLabel(label, false)}
|
||||||
|
</ha-list-item>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</ha-button-menu>`
|
||||||
|
: nothing}
|
||||||
|
</ha-chip-set>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _renderLabel(label: LabelRegistryEntry, clickAction: boolean) {
|
||||||
|
const color = label?.color ? computeCssColor(label.color) : undefined;
|
||||||
|
return html`
|
||||||
|
<ha-label
|
||||||
|
dense
|
||||||
|
role="button"
|
||||||
|
tabindex="0"
|
||||||
|
.item=${label}
|
||||||
|
@click=${clickAction ? this._labelClicked : undefined}
|
||||||
|
@keydown=${clickAction ? this._labelClicked : undefined}
|
||||||
|
style=${color ? `--color: ${color}` : ""}
|
||||||
|
>
|
||||||
|
${label?.icon
|
||||||
|
? html`<ha-icon slot="icon" .icon=${label.icon}></ha-icon>`
|
||||||
|
: nothing}
|
||||||
|
${label.name}
|
||||||
|
</ha-label>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _labelClicked(ev) {
|
||||||
|
ev.stopPropagation();
|
||||||
|
if (ev.type === "keydown" && ev.key !== "Enter" && ev.key !== " ") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const label = (ev.currentTarget as any).item as LabelRegistryEntry;
|
||||||
|
fireEvent(this, "label-clicked", { label });
|
||||||
|
}
|
||||||
|
|
||||||
|
protected _handleIconOverflowMenuOpened(e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
// If this component is used inside a data table, the z-index of the row
|
||||||
|
// needs to be increased. Otherwise the ha-button-menu would be displayed
|
||||||
|
// underneath the next row in the table.
|
||||||
|
const row = this.closest(".mdc-data-table__row") as HTMLDivElement | null;
|
||||||
|
if (row) {
|
||||||
|
row.style.zIndex = "1";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected _handleIconOverflowMenuClosed() {
|
||||||
|
const row = this.closest(".mdc-data-table__row") as HTMLDivElement | null;
|
||||||
|
if (row) {
|
||||||
|
row.style.zIndex = "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles() {
|
||||||
|
return css`
|
||||||
|
:host {
|
||||||
|
display: block;
|
||||||
|
flex-grow: 1;
|
||||||
|
margin-top: 4px;
|
||||||
|
height: 22px;
|
||||||
|
}
|
||||||
|
ha-chip-set {
|
||||||
|
position: fixed;
|
||||||
|
flex-wrap: nowrap;
|
||||||
|
}
|
||||||
|
ha-label {
|
||||||
|
--ha-label-background-color: var(--color, var(--grey-color));
|
||||||
|
--ha-label-background-opacity: 0.5;
|
||||||
|
}
|
||||||
|
ha-button-menu {
|
||||||
|
border-radius: 10px;
|
||||||
|
}
|
||||||
|
.plus {
|
||||||
|
--ha-label-background-color: transparent;
|
||||||
|
border: 1px solid var(--divider-color);
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-data-table-labels": HaDataTableLabels;
|
||||||
|
}
|
||||||
|
interface HASSDomEvents {
|
||||||
|
"label-clicked": { label: LabelRegistryEntry };
|
||||||
|
}
|
||||||
|
}
|
@@ -1,13 +1,13 @@
|
|||||||
import { mdiArrowDown, mdiArrowUp } from "@mdi/js";
|
import { mdiArrowDown, mdiArrowUp, mdiChevronUp } from "@mdi/js";
|
||||||
import deepClone from "deep-clone-simple";
|
import deepClone from "deep-clone-simple";
|
||||||
import {
|
import {
|
||||||
css,
|
|
||||||
CSSResultGroup,
|
CSSResultGroup,
|
||||||
html,
|
|
||||||
LitElement,
|
LitElement,
|
||||||
nothing,
|
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
|
css,
|
||||||
|
html,
|
||||||
|
nothing,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import {
|
import {
|
||||||
customElement,
|
customElement,
|
||||||
@@ -22,7 +22,9 @@ import { styleMap } from "lit/directives/style-map";
|
|||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { restoreScroll } from "../../common/decorators/restore-scroll";
|
import { restoreScroll } from "../../common/decorators/restore-scroll";
|
||||||
import { fireEvent } from "../../common/dom/fire_event";
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
|
import { stringCompare } from "../../common/string/compare";
|
||||||
import { debounce } from "../../common/util/debounce";
|
import { debounce } from "../../common/util/debounce";
|
||||||
|
import { groupBy } from "../../common/util/group-by";
|
||||||
import { nextRender } from "../../common/util/render-status";
|
import { nextRender } from "../../common/util/render-status";
|
||||||
import { haStyleScrollbar } from "../../resources/styles";
|
import { haStyleScrollbar } from "../../resources/styles";
|
||||||
import { loadVirtualizer } from "../../resources/virtualizer";
|
import { loadVirtualizer } from "../../resources/virtualizer";
|
||||||
@@ -33,15 +35,6 @@ import "../ha-svg-icon";
|
|||||||
import "../search-input";
|
import "../search-input";
|
||||||
import { filterData, sortData } from "./sort-filter";
|
import { filterData, sortData } from "./sort-filter";
|
||||||
|
|
||||||
declare global {
|
|
||||||
// for fire event
|
|
||||||
interface HASSDomEvents {
|
|
||||||
"selection-changed": SelectionChangedEvent;
|
|
||||||
"row-click": RowClickedEvent;
|
|
||||||
"sorting-changed": SortingChangedEvent;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface RowClickedEvent {
|
export interface RowClickedEvent {
|
||||||
id: string;
|
id: string;
|
||||||
}
|
}
|
||||||
@@ -50,6 +43,10 @@ export interface SelectionChangedEvent {
|
|||||||
value: string[];
|
value: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface CollapsedChangedEvent {
|
||||||
|
value: string[];
|
||||||
|
}
|
||||||
|
|
||||||
export interface SortingChangedEvent {
|
export interface SortingChangedEvent {
|
||||||
column: string;
|
column: string;
|
||||||
direction: SortingDirection;
|
direction: SortingDirection;
|
||||||
@@ -67,13 +64,20 @@ export interface DataTableSortColumnData {
|
|||||||
filterKey?: string;
|
filterKey?: string;
|
||||||
valueColumn?: string;
|
valueColumn?: string;
|
||||||
direction?: SortingDirection;
|
direction?: SortingDirection;
|
||||||
|
groupable?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DataTableColumnData<T = any> extends DataTableSortColumnData {
|
export interface DataTableColumnData<T = any> extends DataTableSortColumnData {
|
||||||
main?: boolean;
|
main?: boolean;
|
||||||
title: TemplateResult | string;
|
title: TemplateResult | string;
|
||||||
label?: TemplateResult | string;
|
label?: TemplateResult | string;
|
||||||
type?: "numeric" | "icon" | "icon-button" | "overflow-menu" | "flex";
|
type?:
|
||||||
|
| "numeric"
|
||||||
|
| "icon"
|
||||||
|
| "icon-button"
|
||||||
|
| "overflow"
|
||||||
|
| "overflow-menu"
|
||||||
|
| "flex";
|
||||||
template?: (row: T) => TemplateResult | string | typeof nothing;
|
template?: (row: T) => TemplateResult | string | typeof nothing;
|
||||||
width?: string;
|
width?: string;
|
||||||
maxWidth?: string;
|
maxWidth?: string;
|
||||||
@@ -95,6 +99,8 @@ export interface SortableColumnContainer {
|
|||||||
[key: string]: ClonedDataTableColumnData;
|
[key: string]: ClonedDataTableColumnData;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const UNDEFINED_GROUP_KEY = "zzzzz_undefined";
|
||||||
|
|
||||||
@customElement("ha-data-table")
|
@customElement("ha-data-table")
|
||||||
export class HaDataTable extends LitElement {
|
export class HaDataTable extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
@@ -129,14 +135,20 @@ export class HaDataTable extends LitElement {
|
|||||||
|
|
||||||
@property({ type: String }) public filter = "";
|
@property({ type: String }) public filter = "";
|
||||||
|
|
||||||
|
@property() public groupColumn?: string;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public groupOrder?: string[];
|
||||||
|
|
||||||
|
@property() public sortColumn?: string;
|
||||||
|
|
||||||
|
@property() public sortDirection: SortingDirection = null;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public initialCollapsedGroups?: string[];
|
||||||
|
|
||||||
@state() private _filterable = false;
|
@state() private _filterable = false;
|
||||||
|
|
||||||
@state() private _filter = "";
|
@state() private _filter = "";
|
||||||
|
|
||||||
@state() private _sortColumn?: string;
|
|
||||||
|
|
||||||
@state() private _sortDirection: SortingDirection = null;
|
|
||||||
|
|
||||||
@state() private _filteredData: DataTableRowData[] = [];
|
@state() private _filteredData: DataTableRowData[] = [];
|
||||||
|
|
||||||
@state() private _headerHeight = 0;
|
@state() private _headerHeight = 0;
|
||||||
@@ -145,6 +157,8 @@ export class HaDataTable extends LitElement {
|
|||||||
|
|
||||||
@state() private _items: DataTableRowData[] = [];
|
@state() private _items: DataTableRowData[] = [];
|
||||||
|
|
||||||
|
@state() private _collapsedGroups: string[] = [];
|
||||||
|
|
||||||
private _checkableRowsCount?: number;
|
private _checkableRowsCount?: number;
|
||||||
|
|
||||||
private _checkedRows: string[] = [];
|
private _checkedRows: string[] = [];
|
||||||
@@ -169,6 +183,13 @@ export class HaDataTable extends LitElement {
|
|||||||
this._checkedRowsChanged();
|
this._checkedRowsChanged();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public selectAll(): void {
|
||||||
|
this._checkedRows = this._filteredData
|
||||||
|
.filter((data) => data.selectable !== false)
|
||||||
|
.map((data) => data[this.id]);
|
||||||
|
this._checkedRowsChanged();
|
||||||
|
}
|
||||||
|
|
||||||
public connectedCallback() {
|
public connectedCallback() {
|
||||||
super.connectedCallback();
|
super.connectedCallback();
|
||||||
if (this._items.length) {
|
if (this._items.length) {
|
||||||
@@ -193,11 +214,19 @@ export class HaDataTable extends LitElement {
|
|||||||
(column) => column.filterable
|
(column) => column.filterable
|
||||||
);
|
);
|
||||||
|
|
||||||
for (const columnId in this.columns) {
|
if (!this.sortColumn) {
|
||||||
if (this.columns[columnId].direction) {
|
for (const columnId in this.columns) {
|
||||||
this._sortDirection = this.columns[columnId].direction!;
|
if (this.columns[columnId].direction) {
|
||||||
this._sortColumn = columnId;
|
this.sortDirection = this.columns[columnId].direction!;
|
||||||
break;
|
this.sortColumn = columnId;
|
||||||
|
|
||||||
|
fireEvent(this, "sorting-changed", {
|
||||||
|
column: columnId,
|
||||||
|
direction: this.sortDirection,
|
||||||
|
});
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -222,15 +251,30 @@ export class HaDataTable extends LitElement {
|
|||||||
).length;
|
).length;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!this.hasUpdated && this.initialCollapsedGroups) {
|
||||||
|
this._collapsedGroups = this.initialCollapsedGroups;
|
||||||
|
fireEvent(this, "collapsed-changed", { value: this._collapsedGroups });
|
||||||
|
} else if (properties.has("groupColumn")) {
|
||||||
|
this._collapsedGroups = [];
|
||||||
|
fireEvent(this, "collapsed-changed", { value: this._collapsedGroups });
|
||||||
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
properties.has("data") ||
|
properties.has("data") ||
|
||||||
properties.has("columns") ||
|
properties.has("columns") ||
|
||||||
properties.has("_filter") ||
|
properties.has("_filter") ||
|
||||||
properties.has("_sortColumn") ||
|
properties.has("sortColumn") ||
|
||||||
properties.has("_sortDirection")
|
properties.has("sortDirection") ||
|
||||||
|
properties.has("groupColumn") ||
|
||||||
|
properties.has("groupOrder") ||
|
||||||
|
properties.has("_collapsedGroups")
|
||||||
) {
|
) {
|
||||||
this._sortFilterData();
|
this._sortFilterData();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (properties.has("selectable")) {
|
||||||
|
this._items = [...this._items];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render() {
|
protected render() {
|
||||||
@@ -263,75 +307,79 @@ export class HaDataTable extends LitElement {
|
|||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<div class="mdc-data-table__header-row" role="row" aria-rowindex="1">
|
<div class="mdc-data-table__header-row" role="row" aria-rowindex="1">
|
||||||
${this.selectable
|
<slot name="header-row">
|
||||||
? html`
|
${this.selectable
|
||||||
<div
|
? html`
|
||||||
class="mdc-data-table__header-cell mdc-data-table__header-cell--checkbox"
|
<div
|
||||||
role="columnheader"
|
class="mdc-data-table__header-cell mdc-data-table__header-cell--checkbox"
|
||||||
>
|
role="columnheader"
|
||||||
<ha-checkbox
|
|
||||||
class="mdc-data-table__row-checkbox"
|
|
||||||
@change=${this._handleHeaderRowCheckboxClick}
|
|
||||||
.indeterminate=${this._checkedRows.length &&
|
|
||||||
this._checkedRows.length !== this._checkableRowsCount}
|
|
||||||
.checked=${this._checkedRows.length &&
|
|
||||||
this._checkedRows.length === this._checkableRowsCount}
|
|
||||||
>
|
>
|
||||||
</ha-checkbox>
|
<ha-checkbox
|
||||||
|
class="mdc-data-table__row-checkbox"
|
||||||
|
@change=${this._handleHeaderRowCheckboxClick}
|
||||||
|
.indeterminate=${this._checkedRows.length &&
|
||||||
|
this._checkedRows.length !== this._checkableRowsCount}
|
||||||
|
.checked=${this._checkedRows.length &&
|
||||||
|
this._checkedRows.length === this._checkableRowsCount}
|
||||||
|
>
|
||||||
|
</ha-checkbox>
|
||||||
|
</div>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
${Object.entries(this.columns).map(([key, column]) => {
|
||||||
|
if (column.hidden) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
const sorted = key === this.sortColumn;
|
||||||
|
const classes = {
|
||||||
|
"mdc-data-table__header-cell--numeric":
|
||||||
|
column.type === "numeric",
|
||||||
|
"mdc-data-table__header-cell--icon": column.type === "icon",
|
||||||
|
"mdc-data-table__header-cell--icon-button":
|
||||||
|
column.type === "icon-button",
|
||||||
|
"mdc-data-table__header-cell--overflow-menu":
|
||||||
|
column.type === "overflow-menu",
|
||||||
|
"mdc-data-table__header-cell--overflow":
|
||||||
|
column.type === "overflow",
|
||||||
|
sortable: Boolean(column.sortable),
|
||||||
|
"not-sorted": Boolean(column.sortable && !sorted),
|
||||||
|
grows: Boolean(column.grows),
|
||||||
|
};
|
||||||
|
return html`
|
||||||
|
<div
|
||||||
|
aria-label=${ifDefined(column.label)}
|
||||||
|
class="mdc-data-table__header-cell ${classMap(classes)}"
|
||||||
|
style=${column.width
|
||||||
|
? styleMap({
|
||||||
|
[column.grows ? "minWidth" : "width"]: column.width,
|
||||||
|
maxWidth: column.maxWidth || "",
|
||||||
|
})
|
||||||
|
: ""}
|
||||||
|
role="columnheader"
|
||||||
|
aria-sort=${ifDefined(
|
||||||
|
sorted
|
||||||
|
? this.sortDirection === "desc"
|
||||||
|
? "descending"
|
||||||
|
: "ascending"
|
||||||
|
: undefined
|
||||||
|
)}
|
||||||
|
@click=${this._handleHeaderClick}
|
||||||
|
.columnId=${key}
|
||||||
|
>
|
||||||
|
${column.sortable
|
||||||
|
? html`
|
||||||
|
<ha-svg-icon
|
||||||
|
.path=${sorted && this.sortDirection === "desc"
|
||||||
|
? mdiArrowDown
|
||||||
|
: mdiArrowUp}
|
||||||
|
></ha-svg-icon>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
<span>${column.title}</span>
|
||||||
</div>
|
</div>
|
||||||
`
|
`;
|
||||||
: ""}
|
})}
|
||||||
${Object.entries(this.columns).map(([key, column]) => {
|
</slot>
|
||||||
if (column.hidden) {
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
const sorted = key === this._sortColumn;
|
|
||||||
const classes = {
|
|
||||||
"mdc-data-table__header-cell--numeric":
|
|
||||||
column.type === "numeric",
|
|
||||||
"mdc-data-table__header-cell--icon": column.type === "icon",
|
|
||||||
"mdc-data-table__header-cell--icon-button":
|
|
||||||
column.type === "icon-button",
|
|
||||||
"mdc-data-table__header-cell--overflow-menu":
|
|
||||||
column.type === "overflow-menu",
|
|
||||||
sortable: Boolean(column.sortable),
|
|
||||||
"not-sorted": Boolean(column.sortable && !sorted),
|
|
||||||
grows: Boolean(column.grows),
|
|
||||||
};
|
|
||||||
return html`
|
|
||||||
<div
|
|
||||||
aria-label=${ifDefined(column.label)}
|
|
||||||
class="mdc-data-table__header-cell ${classMap(classes)}"
|
|
||||||
style=${column.width
|
|
||||||
? styleMap({
|
|
||||||
[column.grows ? "minWidth" : "width"]: column.width,
|
|
||||||
maxWidth: column.maxWidth || "",
|
|
||||||
})
|
|
||||||
: ""}
|
|
||||||
role="columnheader"
|
|
||||||
aria-sort=${ifDefined(
|
|
||||||
sorted
|
|
||||||
? this._sortDirection === "desc"
|
|
||||||
? "descending"
|
|
||||||
: "ascending"
|
|
||||||
: undefined
|
|
||||||
)}
|
|
||||||
@click=${this._handleHeaderClick}
|
|
||||||
.columnId=${key}
|
|
||||||
>
|
|
||||||
${column.sortable
|
|
||||||
? html`
|
|
||||||
<ha-svg-icon
|
|
||||||
.path=${sorted && this._sortDirection === "desc"
|
|
||||||
? mdiArrowDown
|
|
||||||
: mdiArrowUp}
|
|
||||||
></ha-svg-icon>
|
|
||||||
`
|
|
||||||
: ""}
|
|
||||||
<span>${column.title}</span>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
})}
|
|
||||||
</div>
|
</div>
|
||||||
${!this._filteredData.length
|
${!this._filteredData.length
|
||||||
? html`
|
? html`
|
||||||
@@ -359,7 +407,7 @@ export class HaDataTable extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
private _keyFunction = (row: DataTableRowData) => row[this.id] || row;
|
private _keyFunction = (row: DataTableRowData) => row?.[this.id] || row;
|
||||||
|
|
||||||
private _renderRow = (row: DataTableRowData, index: number) => {
|
private _renderRow = (row: DataTableRowData, index: number) => {
|
||||||
// not sure how this happens...
|
// not sure how this happens...
|
||||||
@@ -408,10 +456,12 @@ export class HaDataTable extends LitElement {
|
|||||||
: ""}
|
: ""}
|
||||||
${Object.entries(this.columns).map(([key, column]) => {
|
${Object.entries(this.columns).map(([key, column]) => {
|
||||||
if (column.hidden) {
|
if (column.hidden) {
|
||||||
return "";
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<div
|
<div
|
||||||
|
@mouseover=${this._setTitle}
|
||||||
|
@focus=${this._setTitle}
|
||||||
role=${column.main ? "rowheader" : "cell"}
|
role=${column.main ? "rowheader" : "cell"}
|
||||||
class="mdc-data-table__cell ${classMap({
|
class="mdc-data-table__cell ${classMap({
|
||||||
"mdc-data-table__cell--flex": column.type === "flex",
|
"mdc-data-table__cell--flex": column.type === "flex",
|
||||||
@@ -421,6 +471,7 @@ export class HaDataTable extends LitElement {
|
|||||||
column.type === "icon-button",
|
column.type === "icon-button",
|
||||||
"mdc-data-table__cell--overflow-menu":
|
"mdc-data-table__cell--overflow-menu":
|
||||||
column.type === "overflow-menu",
|
column.type === "overflow-menu",
|
||||||
|
"mdc-data-table__cell--overflow": column.type === "overflow",
|
||||||
grows: Boolean(column.grows),
|
grows: Boolean(column.grows),
|
||||||
forceLTR: Boolean(column.forceLTR),
|
forceLTR: Boolean(column.forceLTR),
|
||||||
})}"
|
})}"
|
||||||
@@ -453,12 +504,12 @@ export class HaDataTable extends LitElement {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const prom = this._sortColumn
|
const prom = this.sortColumn
|
||||||
? sortData(
|
? sortData(
|
||||||
filteredData,
|
filteredData,
|
||||||
this._sortColumns[this._sortColumn],
|
this._sortColumns[this.sortColumn],
|
||||||
this._sortDirection,
|
this.sortDirection,
|
||||||
this._sortColumn,
|
this.sortColumn,
|
||||||
this.hass.locale.language
|
this.hass.locale.language
|
||||||
)
|
)
|
||||||
: filteredData;
|
: filteredData;
|
||||||
@@ -477,8 +528,75 @@ export class HaDataTable extends LitElement {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.appendRow || this.hasFab) {
|
if (this.appendRow || this.hasFab || this.groupColumn) {
|
||||||
const items = [...data];
|
let items = [...data];
|
||||||
|
|
||||||
|
if (this.groupColumn) {
|
||||||
|
const grouped = groupBy(items, (item) => item[this.groupColumn!]);
|
||||||
|
if (grouped.undefined) {
|
||||||
|
// make sure ungrouped items are at the bottom
|
||||||
|
grouped[UNDEFINED_GROUP_KEY] = grouped.undefined;
|
||||||
|
delete grouped.undefined;
|
||||||
|
}
|
||||||
|
const sorted: {
|
||||||
|
[key: string]: DataTableRowData[];
|
||||||
|
} = Object.keys(grouped)
|
||||||
|
.sort((a, b) => {
|
||||||
|
const orderA = this.groupOrder?.indexOf(a) ?? -1;
|
||||||
|
const orderB = this.groupOrder?.indexOf(b) ?? -1;
|
||||||
|
if (orderA !== orderB) {
|
||||||
|
if (orderA === -1) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
if (orderB === -1) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
return orderA - orderB;
|
||||||
|
}
|
||||||
|
return stringCompare(
|
||||||
|
["", "-", "—"].includes(a) ? "zzz" : a,
|
||||||
|
["", "-", "—"].includes(b) ? "zzz" : b,
|
||||||
|
this.hass.locale.language
|
||||||
|
);
|
||||||
|
})
|
||||||
|
.reduce((obj, key) => {
|
||||||
|
obj[key] = grouped[key];
|
||||||
|
return obj;
|
||||||
|
}, {});
|
||||||
|
const groupedItems: DataTableRowData[] = [];
|
||||||
|
Object.entries(sorted).forEach(([groupName, rows]) => {
|
||||||
|
if (
|
||||||
|
groupName !== UNDEFINED_GROUP_KEY ||
|
||||||
|
Object.keys(sorted).length > 1
|
||||||
|
) {
|
||||||
|
groupedItems.push({
|
||||||
|
append: true,
|
||||||
|
content: html`<div
|
||||||
|
class="mdc-data-table__cell group-header"
|
||||||
|
role="cell"
|
||||||
|
.group=${groupName}
|
||||||
|
@click=${this._collapseGroup}
|
||||||
|
>
|
||||||
|
<ha-icon-button
|
||||||
|
.path=${mdiChevronUp}
|
||||||
|
class=${this._collapsedGroups.includes(groupName)
|
||||||
|
? "collapsed"
|
||||||
|
: ""}
|
||||||
|
>
|
||||||
|
</ha-icon-button>
|
||||||
|
${groupName === UNDEFINED_GROUP_KEY
|
||||||
|
? this.hass.localize("ui.components.data-table.ungrouped")
|
||||||
|
: groupName || ""}
|
||||||
|
</div>`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (!this._collapsedGroups.includes(groupName)) {
|
||||||
|
groupedItems.push(...rows);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
items = groupedItems;
|
||||||
|
}
|
||||||
|
|
||||||
if (this.appendRow) {
|
if (this.appendRow) {
|
||||||
items.push({ append: true, content: this.appendRow });
|
items.push({ append: true, content: this.appendRow });
|
||||||
@@ -487,6 +605,7 @@ export class HaDataTable extends LitElement {
|
|||||||
if (this.hasFab) {
|
if (this.hasFab) {
|
||||||
items.push({ empty: true });
|
items.push({ empty: true });
|
||||||
}
|
}
|
||||||
|
|
||||||
this._items = items;
|
this._items = items;
|
||||||
} else {
|
} else {
|
||||||
this._items = data;
|
this._items = data;
|
||||||
@@ -507,29 +626,26 @@ export class HaDataTable extends LitElement {
|
|||||||
if (!this.columns[columnId].sortable) {
|
if (!this.columns[columnId].sortable) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!this._sortDirection || this._sortColumn !== columnId) {
|
if (!this.sortDirection || this.sortColumn !== columnId) {
|
||||||
this._sortDirection = "asc";
|
this.sortDirection = "asc";
|
||||||
} else if (this._sortDirection === "asc") {
|
} else if (this.sortDirection === "asc") {
|
||||||
this._sortDirection = "desc";
|
this.sortDirection = "desc";
|
||||||
} else {
|
} else {
|
||||||
this._sortDirection = null;
|
this.sortDirection = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
this._sortColumn = this._sortDirection === null ? undefined : columnId;
|
this.sortColumn = this.sortDirection === null ? undefined : columnId;
|
||||||
|
|
||||||
fireEvent(this, "sorting-changed", {
|
fireEvent(this, "sorting-changed", {
|
||||||
column: columnId,
|
column: columnId,
|
||||||
direction: this._sortDirection,
|
direction: this.sortDirection,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private _handleHeaderRowCheckboxClick(ev: Event) {
|
private _handleHeaderRowCheckboxClick(ev: Event) {
|
||||||
const checkbox = ev.target as HaCheckbox;
|
const checkbox = ev.target as HaCheckbox;
|
||||||
if (checkbox.checked) {
|
if (checkbox.checked) {
|
||||||
this._checkedRows = this._filteredData
|
this.selectAll();
|
||||||
.filter((data) => data.selectable !== false)
|
|
||||||
.map((data) => data[this.id]);
|
|
||||||
this._checkedRowsChanged();
|
|
||||||
} else {
|
} else {
|
||||||
this._checkedRows = [];
|
this._checkedRows = [];
|
||||||
this._checkedRowsChanged();
|
this._checkedRowsChanged();
|
||||||
@@ -552,14 +668,32 @@ export class HaDataTable extends LitElement {
|
|||||||
};
|
};
|
||||||
|
|
||||||
private _handleRowClick = (ev: Event) => {
|
private _handleRowClick = (ev: Event) => {
|
||||||
const target = ev.target as HTMLElement;
|
if (
|
||||||
if (["HA-CHECKBOX", "MWC-BUTTON"].includes(target.tagName)) {
|
ev
|
||||||
|
.composedPath()
|
||||||
|
.find((el) =>
|
||||||
|
[
|
||||||
|
"ha-checkbox",
|
||||||
|
"mwc-button",
|
||||||
|
"ha-button",
|
||||||
|
"ha-icon-button",
|
||||||
|
"ha-assist-chip",
|
||||||
|
].includes((el as HTMLElement).localName)
|
||||||
|
)
|
||||||
|
) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const rowId = (ev.currentTarget as any).rowId;
|
const rowId = (ev.currentTarget as any).rowId;
|
||||||
fireEvent(this, "row-click", { id: rowId }, { bubbles: false });
|
fireEvent(this, "row-click", { id: rowId }, { bubbles: false });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
private _setTitle(ev: Event) {
|
||||||
|
const target = ev.currentTarget as HTMLElement;
|
||||||
|
if (target.scrollWidth > target.offsetWidth) {
|
||||||
|
target.setAttribute("title", target.innerText);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private _checkedRowsChanged() {
|
private _checkedRowsChanged() {
|
||||||
// force scroller to update, change it's items
|
// force scroller to update, change it's items
|
||||||
if (this._items.length) {
|
if (this._items.length) {
|
||||||
@@ -590,6 +724,18 @@ export class HaDataTable extends LitElement {
|
|||||||
this._savedScrollPos = (e.target as HTMLDivElement).scrollTop;
|
this._savedScrollPos = (e.target as HTMLDivElement).scrollTop;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private _collapseGroup = (ev: Event) => {
|
||||||
|
const groupName = (ev.currentTarget as any).group;
|
||||||
|
if (this._collapsedGroups.includes(groupName)) {
|
||||||
|
this._collapsedGroups = this._collapsedGroups.filter(
|
||||||
|
(grp) => grp !== groupName
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
this._collapsedGroups = [...this._collapsedGroups, groupName];
|
||||||
|
}
|
||||||
|
fireEvent(this, "collapsed-changed", { value: this._collapsedGroups });
|
||||||
|
};
|
||||||
|
|
||||||
static get styles(): CSSResultGroup {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyleScrollbar,
|
haStyleScrollbar,
|
||||||
@@ -629,7 +775,7 @@ export class HaDataTable extends LitElement {
|
|||||||
.mdc-data-table__row {
|
.mdc-data-table__row {
|
||||||
display: flex;
|
display: flex;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 52px;
|
height: var(--data-table-row-height, 52px);
|
||||||
}
|
}
|
||||||
|
|
||||||
.mdc-data-table__row ~ .mdc-data-table__row {
|
.mdc-data-table__row ~ .mdc-data-table__row {
|
||||||
@@ -655,7 +801,6 @@ export class HaDataTable extends LitElement {
|
|||||||
display: flex;
|
display: flex;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
border-bottom: 1px solid var(--divider-color);
|
border-bottom: 1px solid var(--divider-color);
|
||||||
overflow-x: auto;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.mdc-data-table__header-row::-webkit-scrollbar {
|
.mdc-data-table__header-row::-webkit-scrollbar {
|
||||||
@@ -809,7 +954,9 @@ export class HaDataTable extends LitElement {
|
|||||||
padding-inline-start: initial;
|
padding-inline-start: initial;
|
||||||
}
|
}
|
||||||
.mdc-data-table__cell--overflow-menu,
|
.mdc-data-table__cell--overflow-menu,
|
||||||
.mdc-data-table__header-cell--overflow-menu {
|
.mdc-data-table__cell--overflow,
|
||||||
|
.mdc-data-table__header-cell--overflow-menu,
|
||||||
|
.mdc-data-table__header-cell--overflow {
|
||||||
overflow: initial;
|
overflow: initial;
|
||||||
}
|
}
|
||||||
.mdc-data-table__cell--icon-button a {
|
.mdc-data-table__cell--icon-button a {
|
||||||
@@ -839,6 +986,25 @@ export class HaDataTable extends LitElement {
|
|||||||
|
|
||||||
/* custom from here */
|
/* custom from here */
|
||||||
|
|
||||||
|
.group-header {
|
||||||
|
padding-top: 12px;
|
||||||
|
padding-left: 12px;
|
||||||
|
padding-inline-start: 12px;
|
||||||
|
width: 100%;
|
||||||
|
font-weight: 500;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-header ha-icon-button {
|
||||||
|
transition: transform 0.2s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-header ha-icon-button.collapsed {
|
||||||
|
transform: rotate(180deg);
|
||||||
|
}
|
||||||
|
|
||||||
:host {
|
:host {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
@@ -935,4 +1101,12 @@ declare global {
|
|||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
"ha-data-table": HaDataTable;
|
"ha-data-table": HaDataTable;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// for fire event
|
||||||
|
interface HASSDomEvents {
|
||||||
|
"selection-changed": SelectionChangedEvent;
|
||||||
|
"row-click": RowClickedEvent;
|
||||||
|
"sorting-changed": SortingChangedEvent;
|
||||||
|
"collapsed-changed": CollapsedChangedEvent;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -11,10 +11,10 @@ import {
|
|||||||
} from "../common/datetime/localize_date";
|
} from "../common/datetime/localize_date";
|
||||||
import { mainWindow } from "../common/dom/get_main_window";
|
import { mainWindow } from "../common/dom/get_main_window";
|
||||||
|
|
||||||
// Set the current date to the left picker instead of the right picker because the right is hidden
|
|
||||||
const CustomDateRangePicker = Vue.extend({
|
const CustomDateRangePicker = Vue.extend({
|
||||||
mixins: [DateRangePicker],
|
mixins: [DateRangePicker],
|
||||||
methods: {
|
methods: {
|
||||||
|
// Set the current date to the left picker instead of the right picker because the right is hidden
|
||||||
selectMonthDate() {
|
selectMonthDate() {
|
||||||
const dt: Date = this.end || new Date();
|
const dt: Date = this.end || new Date();
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
@@ -23,6 +23,33 @@ const CustomDateRangePicker = Vue.extend({
|
|||||||
month: dt.getMonth() + 1,
|
month: dt.getMonth() + 1,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
// Fix the start/end date calculation when selecting a date range. The
|
||||||
|
// original code keeps track of the first clicked date (in_selection) but it
|
||||||
|
// never sets it to either the start or end date variables, so if the
|
||||||
|
// in_selection date is between the start and end date that were set by the
|
||||||
|
// hover the selection will enter a broken state that's counter-intuitive
|
||||||
|
// when hovering between weeks and leads to a random date when selecting a
|
||||||
|
// range across months. This bug doesn't seem to be present on v0.6.7 of the
|
||||||
|
// lib
|
||||||
|
hoverDate(value: Date) {
|
||||||
|
if (this.readonly) return;
|
||||||
|
|
||||||
|
if (this.in_selection) {
|
||||||
|
const pickA = this.in_selection as Date;
|
||||||
|
const pickB = value;
|
||||||
|
|
||||||
|
this.start = this.normalizeDatetime(
|
||||||
|
Math.min(pickA.valueOf(), pickB.valueOf()),
|
||||||
|
this.start
|
||||||
|
);
|
||||||
|
this.end = this.normalizeDatetime(
|
||||||
|
Math.max(pickA.valueOf(), pickB.valueOf()),
|
||||||
|
this.end
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.$emit("hover-date", value);
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -18,6 +18,12 @@ import "../ha-icon-button";
|
|||||||
import "../ha-svg-icon";
|
import "../ha-svg-icon";
|
||||||
import "./state-badge";
|
import "./state-badge";
|
||||||
import { caseInsensitiveStringCompare } from "../../common/string/compare";
|
import { caseInsensitiveStringCompare } from "../../common/string/compare";
|
||||||
|
import { showHelperDetailDialog } from "../../panels/config/helpers/show-dialog-helper-detail";
|
||||||
|
import { domainToName } from "../../data/integration";
|
||||||
|
import {
|
||||||
|
isHelperDomain,
|
||||||
|
HelperDomain,
|
||||||
|
} from "../../panels/config/helpers/const";
|
||||||
|
|
||||||
interface HassEntityWithCachedName extends HassEntity, ScorableTextItem {
|
interface HassEntityWithCachedName extends HassEntity, ScorableTextItem {
|
||||||
friendly_name: string;
|
friendly_name: string;
|
||||||
@@ -25,6 +31,8 @@ interface HassEntityWithCachedName extends HassEntity, ScorableTextItem {
|
|||||||
|
|
||||||
export type HaEntityPickerEntityFilterFunc = (entity: HassEntity) => boolean;
|
export type HaEntityPickerEntityFilterFunc = (entity: HassEntity) => boolean;
|
||||||
|
|
||||||
|
const CREATE_ID = "___create-new-entity___";
|
||||||
|
|
||||||
@customElement("ha-entity-picker")
|
@customElement("ha-entity-picker")
|
||||||
export class HaEntityPicker extends LitElement {
|
export class HaEntityPicker extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
@@ -44,6 +52,8 @@ export class HaEntityPicker extends LitElement {
|
|||||||
|
|
||||||
@property() public helper?: string;
|
@property() public helper?: string;
|
||||||
|
|
||||||
|
@property({ type: Array }) public createDomains?: string[];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Show entities from specific domains.
|
* Show entities from specific domains.
|
||||||
* @type {Array}
|
* @type {Array}
|
||||||
@@ -130,7 +140,11 @@ export class HaEntityPicker extends LitElement {
|
|||||||
></state-badge>`
|
></state-badge>`
|
||||||
: ""}
|
: ""}
|
||||||
<span>${item.friendly_name}</span>
|
<span>${item.friendly_name}</span>
|
||||||
<span slot="secondary">${item.entity_id}</span>
|
<span slot="secondary"
|
||||||
|
>${item.entity_id.startsWith(CREATE_ID)
|
||||||
|
? this.hass.localize("ui.components.entity.entity-picker.new_entity")
|
||||||
|
: item.entity_id}</span
|
||||||
|
>
|
||||||
</ha-list-item>`;
|
</ha-list-item>`;
|
||||||
|
|
||||||
private _getStates = memoizeOne(
|
private _getStates = memoizeOne(
|
||||||
@@ -143,7 +157,8 @@ export class HaEntityPicker extends LitElement {
|
|||||||
includeDeviceClasses: this["includeDeviceClasses"],
|
includeDeviceClasses: this["includeDeviceClasses"],
|
||||||
includeUnitOfMeasurement: this["includeUnitOfMeasurement"],
|
includeUnitOfMeasurement: this["includeUnitOfMeasurement"],
|
||||||
includeEntities: this["includeEntities"],
|
includeEntities: this["includeEntities"],
|
||||||
excludeEntities: this["excludeEntities"]
|
excludeEntities: this["excludeEntities"],
|
||||||
|
createDomains: this["createDomains"]
|
||||||
): HassEntityWithCachedName[] => {
|
): HassEntityWithCachedName[] => {
|
||||||
let states: HassEntityWithCachedName[] = [];
|
let states: HassEntityWithCachedName[] = [];
|
||||||
|
|
||||||
@@ -152,6 +167,34 @@ export class HaEntityPicker extends LitElement {
|
|||||||
}
|
}
|
||||||
let entityIds = Object.keys(hass.states);
|
let entityIds = Object.keys(hass.states);
|
||||||
|
|
||||||
|
const createItems = createDomains?.length
|
||||||
|
? createDomains.map((domain) => {
|
||||||
|
const newFriendlyName = hass.localize(
|
||||||
|
"ui.components.entity.entity-picker.create_helper",
|
||||||
|
{
|
||||||
|
domain: isHelperDomain(domain)
|
||||||
|
? hass.localize(
|
||||||
|
`ui.panel.config.helpers.types.${domain as HelperDomain}`
|
||||||
|
)
|
||||||
|
: domainToName(hass.localize, domain),
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
entity_id: CREATE_ID + domain,
|
||||||
|
state: "on",
|
||||||
|
last_changed: "",
|
||||||
|
last_updated: "",
|
||||||
|
context: { id: "", user_id: null, parent_id: null },
|
||||||
|
friendly_name: newFriendlyName,
|
||||||
|
attributes: {
|
||||||
|
icon: "mdi:plus",
|
||||||
|
},
|
||||||
|
strings: [domain, newFriendlyName],
|
||||||
|
};
|
||||||
|
})
|
||||||
|
: [];
|
||||||
|
|
||||||
if (!entityIds.length) {
|
if (!entityIds.length) {
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
@@ -171,6 +214,7 @@ export class HaEntityPicker extends LitElement {
|
|||||||
},
|
},
|
||||||
strings: [],
|
strings: [],
|
||||||
},
|
},
|
||||||
|
...createItems,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -281,9 +325,14 @@ export class HaEntityPicker extends LitElement {
|
|||||||
},
|
},
|
||||||
strings: [],
|
strings: [],
|
||||||
},
|
},
|
||||||
|
...createItems,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (createItems?.length) {
|
||||||
|
states.push(...createItems);
|
||||||
|
}
|
||||||
|
|
||||||
return states;
|
return states;
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@@ -310,13 +359,18 @@ export class HaEntityPicker extends LitElement {
|
|||||||
this.includeDeviceClasses,
|
this.includeDeviceClasses,
|
||||||
this.includeUnitOfMeasurement,
|
this.includeUnitOfMeasurement,
|
||||||
this.includeEntities,
|
this.includeEntities,
|
||||||
this.excludeEntities
|
this.excludeEntities,
|
||||||
|
this.createDomains
|
||||||
);
|
);
|
||||||
if (this._initedStates) {
|
if (this._initedStates) {
|
||||||
this.comboBox.filteredItems = this._states;
|
this.comboBox.filteredItems = this._states;
|
||||||
}
|
}
|
||||||
this._initedStates = true;
|
this._initedStates = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (changedProps.has("createDomains") && this.createDomains?.length) {
|
||||||
|
this.hass.loadFragmentTranslation("config");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
@@ -354,6 +408,18 @@ export class HaEntityPicker extends LitElement {
|
|||||||
private _valueChanged(ev: ValueChangedEvent<string>) {
|
private _valueChanged(ev: ValueChangedEvent<string>) {
|
||||||
ev.stopPropagation();
|
ev.stopPropagation();
|
||||||
const newValue = ev.detail.value;
|
const newValue = ev.detail.value;
|
||||||
|
|
||||||
|
if (newValue && newValue.startsWith(CREATE_ID)) {
|
||||||
|
const domain = newValue.substring(CREATE_ID.length);
|
||||||
|
showHelperDetailDialog(this, {
|
||||||
|
domain,
|
||||||
|
dialogClosedCallback: (item) => {
|
||||||
|
if (item.entityId) this._setValue(item.entityId);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (newValue !== this._value) {
|
if (newValue !== this._value) {
|
||||||
this._setValue(newValue);
|
this._setValue(newValue);
|
||||||
}
|
}
|
||||||
|
@@ -32,7 +32,9 @@ export class StateBadge extends LitElement {
|
|||||||
|
|
||||||
@property() public overrideImage?: string;
|
@property() public overrideImage?: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public stateColor = false;
|
// Cannot be a boolean attribute because undefined is treated different than
|
||||||
|
// false. When it is undefined, state is still colored for light entities.
|
||||||
|
@property({ attribute: false }) public stateColor?: boolean;
|
||||||
|
|
||||||
@property() public color?: string;
|
@property() public color?: string;
|
||||||
|
|
||||||
@@ -70,7 +72,7 @@ export class StateBadge extends LitElement {
|
|||||||
const domain = this.stateObj
|
const domain = this.stateObj
|
||||||
? computeStateDomain(this.stateObj)
|
? computeStateDomain(this.stateObj)
|
||||||
: undefined;
|
: undefined;
|
||||||
return this.stateColor || (domain === "light" && this.stateColor !== false);
|
return this.stateColor ?? domain === "light";
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render() {
|
protected render() {
|
||||||
|
@@ -1,12 +1,12 @@
|
|||||||
import { mdiSofa } from "@mdi/js";
|
import { mdiTextureBox } from "@mdi/js";
|
||||||
import { CSSResultGroup, LitElement, TemplateResult, css, html } from "lit";
|
import { CSSResultGroup, LitElement, TemplateResult, css, html } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
import { showAreaFilterDialog } from "../dialogs/area-filter/show-area-filter-dialog";
|
import { showAreaFilterDialog } from "../dialogs/area-filter/show-area-filter-dialog";
|
||||||
import { HomeAssistant } from "../types";
|
import { HomeAssistant } from "../types";
|
||||||
|
import "./ha-icon-next";
|
||||||
import "./ha-svg-icon";
|
import "./ha-svg-icon";
|
||||||
import "./ha-textfield";
|
import "./ha-textfield";
|
||||||
import "./ha-icon-next";
|
|
||||||
|
|
||||||
export type AreaFilterValue = {
|
export type AreaFilterValue = {
|
||||||
hidden?: string[];
|
hidden?: string[];
|
||||||
@@ -51,7 +51,7 @@ export class HaAreaPicker extends LitElement {
|
|||||||
@keydown=${this._edit}
|
@keydown=${this._edit}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
>
|
>
|
||||||
<ha-svg-icon slot="graphic" .path=${mdiSofa}></ha-svg-icon>
|
<ha-svg-icon slot="graphic" .path=${mdiTextureBox}></ha-svg-icon>
|
||||||
<span>${this.label}</span>
|
<span>${this.label}</span>
|
||||||
<span slot="secondary">${description}</span>
|
<span slot="secondary">${description}</span>
|
||||||
<ha-icon-next
|
<ha-icon-next
|
||||||
|
529
src/components/ha-area-floor-picker.ts
Normal file
529
src/components/ha-area-floor-picker.ts
Normal file
@@ -0,0 +1,529 @@
|
|||||||
|
import { mdiTextureBox } from "@mdi/js";
|
||||||
|
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
||||||
|
import { HassEntity, UnsubscribeFunc } from "home-assistant-js-websocket";
|
||||||
|
import { LitElement, PropertyValues, TemplateResult, html, nothing } from "lit";
|
||||||
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
|
import { styleMap } from "lit/directives/style-map";
|
||||||
|
import memoizeOne from "memoize-one";
|
||||||
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
|
import { computeDomain } from "../common/entity/compute_domain";
|
||||||
|
import { stringCompare } from "../common/string/compare";
|
||||||
|
import {
|
||||||
|
ScorableTextItem,
|
||||||
|
fuzzyFilterSort,
|
||||||
|
} from "../common/string/filter/sequence-matching";
|
||||||
|
import { computeRTL } from "../common/util/compute_rtl";
|
||||||
|
import { AreaRegistryEntry } from "../data/area_registry";
|
||||||
|
import {
|
||||||
|
DeviceEntityDisplayLookup,
|
||||||
|
DeviceRegistryEntry,
|
||||||
|
getDeviceEntityDisplayLookup,
|
||||||
|
} from "../data/device_registry";
|
||||||
|
import { EntityRegistryDisplayEntry } from "../data/entity_registry";
|
||||||
|
import {
|
||||||
|
FloorRegistryEntry,
|
||||||
|
getFloorAreaLookup,
|
||||||
|
subscribeFloorRegistry,
|
||||||
|
} from "../data/floor_registry";
|
||||||
|
import { SubscribeMixin } from "../mixins/subscribe-mixin";
|
||||||
|
import { HomeAssistant, ValueChangedEvent } from "../types";
|
||||||
|
import type { HaDevicePickerDeviceFilterFunc } from "./device/ha-device-picker";
|
||||||
|
import "./ha-combo-box";
|
||||||
|
import type { HaComboBox } from "./ha-combo-box";
|
||||||
|
import "./ha-floor-icon";
|
||||||
|
import "./ha-icon-button";
|
||||||
|
import "./ha-list-item";
|
||||||
|
import "./ha-svg-icon";
|
||||||
|
import "./ha-tree-indicator";
|
||||||
|
|
||||||
|
type ScorableAreaFloorEntry = ScorableTextItem & FloorAreaEntry;
|
||||||
|
|
||||||
|
interface FloorAreaEntry {
|
||||||
|
id: string | null;
|
||||||
|
name: string;
|
||||||
|
icon: string | null;
|
||||||
|
strings: string[];
|
||||||
|
type: "floor" | "area";
|
||||||
|
level: number | null;
|
||||||
|
hasFloor?: boolean;
|
||||||
|
lastArea?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
@customElement("ha-area-floor-picker")
|
||||||
|
export class HaAreaFloorPicker extends SubscribeMixin(LitElement) {
|
||||||
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public value?: string;
|
||||||
|
|
||||||
|
@property() public helper?: string;
|
||||||
|
|
||||||
|
@property() public placeholder?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Show only areas with entities from specific domains.
|
||||||
|
* @type {Array}
|
||||||
|
* @attr include-domains
|
||||||
|
*/
|
||||||
|
@property({ type: Array, attribute: "include-domains" })
|
||||||
|
public includeDomains?: string[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Show no areas with entities of these domains.
|
||||||
|
* @type {Array}
|
||||||
|
* @attr exclude-domains
|
||||||
|
*/
|
||||||
|
@property({ type: Array, attribute: "exclude-domains" })
|
||||||
|
public excludeDomains?: string[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Show only areas with entities of these device classes.
|
||||||
|
* @type {Array}
|
||||||
|
* @attr include-device-classes
|
||||||
|
*/
|
||||||
|
@property({ type: Array, attribute: "include-device-classes" })
|
||||||
|
public includeDeviceClasses?: string[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List of areas to be excluded.
|
||||||
|
* @type {Array}
|
||||||
|
* @attr exclude-areas
|
||||||
|
*/
|
||||||
|
@property({ type: Array, attribute: "exclude-areas" })
|
||||||
|
public excludeAreas?: string[];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List of floors to be excluded.
|
||||||
|
* @type {Array}
|
||||||
|
* @attr exclude-floors
|
||||||
|
*/
|
||||||
|
@property({ type: Array, attribute: "exclude-floors" })
|
||||||
|
public excludeFloors?: string[];
|
||||||
|
|
||||||
|
@property({ attribute: false })
|
||||||
|
public deviceFilter?: HaDevicePickerDeviceFilterFunc;
|
||||||
|
|
||||||
|
@property({ attribute: false })
|
||||||
|
public entityFilter?: (entity: HassEntity) => boolean;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public required = false;
|
||||||
|
|
||||||
|
@state() private _floors?: FloorRegistryEntry[];
|
||||||
|
|
||||||
|
@state() private _opened?: boolean;
|
||||||
|
|
||||||
|
@query("ha-combo-box", true) public comboBox!: HaComboBox;
|
||||||
|
|
||||||
|
private _init = false;
|
||||||
|
|
||||||
|
protected hassSubscribe(): (UnsubscribeFunc | Promise<UnsubscribeFunc>)[] {
|
||||||
|
return [
|
||||||
|
subscribeFloorRegistry(this.hass.connection, (floors) => {
|
||||||
|
this._floors = floors;
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public async open() {
|
||||||
|
await this.updateComplete;
|
||||||
|
await this.comboBox?.open();
|
||||||
|
}
|
||||||
|
|
||||||
|
public async focus() {
|
||||||
|
await this.updateComplete;
|
||||||
|
await this.comboBox?.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
private _rowRenderer: ComboBoxLitRenderer<FloorAreaEntry> = (item) => {
|
||||||
|
const rtl = computeRTL(this.hass);
|
||||||
|
return html`
|
||||||
|
<ha-list-item
|
||||||
|
graphic="icon"
|
||||||
|
style=${item.type === "area" && item.hasFloor
|
||||||
|
? rtl
|
||||||
|
? "--mdc-list-side-padding-right: 48px;"
|
||||||
|
: "--mdc-list-side-padding-left: 48px;"
|
||||||
|
: ""}
|
||||||
|
>
|
||||||
|
${item.type === "area" && item.hasFloor
|
||||||
|
? html`<ha-tree-indicator
|
||||||
|
style=${styleMap({
|
||||||
|
width: "48px",
|
||||||
|
position: "absolute",
|
||||||
|
top: "0px",
|
||||||
|
left: rtl ? undefined : "8px",
|
||||||
|
right: rtl ? "8px" : undefined,
|
||||||
|
transform: rtl ? "scaleX(-1)" : "",
|
||||||
|
})}
|
||||||
|
.end=${item.lastArea}
|
||||||
|
slot="graphic"
|
||||||
|
></ha-tree-indicator>`
|
||||||
|
: nothing}
|
||||||
|
${item.type === "floor"
|
||||||
|
? html`<ha-floor-icon slot="graphic" .floor=${item}></ha-floor-icon>`
|
||||||
|
: item.icon
|
||||||
|
? html`<ha-icon slot="graphic" .icon=${item.icon}></ha-icon>`
|
||||||
|
: html`<ha-svg-icon
|
||||||
|
slot="graphic"
|
||||||
|
.path=${mdiTextureBox}
|
||||||
|
></ha-svg-icon>`}
|
||||||
|
${item.name}
|
||||||
|
</ha-list-item>
|
||||||
|
`;
|
||||||
|
};
|
||||||
|
|
||||||
|
private _getAreas = memoizeOne(
|
||||||
|
(
|
||||||
|
floors: FloorRegistryEntry[],
|
||||||
|
areas: AreaRegistryEntry[],
|
||||||
|
devices: DeviceRegistryEntry[],
|
||||||
|
entities: EntityRegistryDisplayEntry[],
|
||||||
|
includeDomains: this["includeDomains"],
|
||||||
|
excludeDomains: this["excludeDomains"],
|
||||||
|
includeDeviceClasses: this["includeDeviceClasses"],
|
||||||
|
deviceFilter: this["deviceFilter"],
|
||||||
|
entityFilter: this["entityFilter"],
|
||||||
|
excludeAreas: this["excludeAreas"],
|
||||||
|
excludeFloors: this["excludeFloors"]
|
||||||
|
): FloorAreaEntry[] => {
|
||||||
|
if (!areas.length && !floors.length) {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: "no_areas",
|
||||||
|
type: "area",
|
||||||
|
name: this.hass.localize("ui.components.area-picker.no_areas"),
|
||||||
|
icon: null,
|
||||||
|
strings: [],
|
||||||
|
level: null,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
let deviceEntityLookup: DeviceEntityDisplayLookup = {};
|
||||||
|
let inputDevices: DeviceRegistryEntry[] | undefined;
|
||||||
|
let inputEntities: EntityRegistryDisplayEntry[] | undefined;
|
||||||
|
|
||||||
|
if (
|
||||||
|
includeDomains ||
|
||||||
|
excludeDomains ||
|
||||||
|
includeDeviceClasses ||
|
||||||
|
deviceFilter ||
|
||||||
|
entityFilter
|
||||||
|
) {
|
||||||
|
deviceEntityLookup = getDeviceEntityDisplayLookup(entities);
|
||||||
|
inputDevices = devices;
|
||||||
|
inputEntities = entities.filter((entity) => entity.area_id);
|
||||||
|
|
||||||
|
if (includeDomains) {
|
||||||
|
inputDevices = inputDevices!.filter((device) => {
|
||||||
|
const devEntities = deviceEntityLookup[device.id];
|
||||||
|
if (!devEntities || !devEntities.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return deviceEntityLookup[device.id].some((entity) =>
|
||||||
|
includeDomains.includes(computeDomain(entity.entity_id))
|
||||||
|
);
|
||||||
|
});
|
||||||
|
inputEntities = inputEntities!.filter((entity) =>
|
||||||
|
includeDomains.includes(computeDomain(entity.entity_id))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (excludeDomains) {
|
||||||
|
inputDevices = inputDevices!.filter((device) => {
|
||||||
|
const devEntities = deviceEntityLookup[device.id];
|
||||||
|
if (!devEntities || !devEntities.length) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return entities.every(
|
||||||
|
(entity) =>
|
||||||
|
!excludeDomains.includes(computeDomain(entity.entity_id))
|
||||||
|
);
|
||||||
|
});
|
||||||
|
inputEntities = inputEntities!.filter(
|
||||||
|
(entity) =>
|
||||||
|
!excludeDomains.includes(computeDomain(entity.entity_id))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (includeDeviceClasses) {
|
||||||
|
inputDevices = inputDevices!.filter((device) => {
|
||||||
|
const devEntities = deviceEntityLookup[device.id];
|
||||||
|
if (!devEntities || !devEntities.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return deviceEntityLookup[device.id].some((entity) => {
|
||||||
|
const stateObj = this.hass.states[entity.entity_id];
|
||||||
|
if (!stateObj) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
stateObj.attributes.device_class &&
|
||||||
|
includeDeviceClasses.includes(stateObj.attributes.device_class)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
inputEntities = inputEntities!.filter((entity) => {
|
||||||
|
const stateObj = this.hass.states[entity.entity_id];
|
||||||
|
return (
|
||||||
|
stateObj.attributes.device_class &&
|
||||||
|
includeDeviceClasses.includes(stateObj.attributes.device_class)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deviceFilter) {
|
||||||
|
inputDevices = inputDevices!.filter((device) =>
|
||||||
|
deviceFilter!(device)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (entityFilter) {
|
||||||
|
inputDevices = inputDevices!.filter((device) => {
|
||||||
|
const devEntities = deviceEntityLookup[device.id];
|
||||||
|
if (!devEntities || !devEntities.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return deviceEntityLookup[device.id].some((entity) => {
|
||||||
|
const stateObj = this.hass.states[entity.entity_id];
|
||||||
|
if (!stateObj) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return entityFilter(stateObj);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
inputEntities = inputEntities!.filter((entity) => {
|
||||||
|
const stateObj = this.hass.states[entity.entity_id];
|
||||||
|
if (!stateObj) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return entityFilter!(stateObj);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let outputAreas = areas;
|
||||||
|
|
||||||
|
let areaIds: string[] | undefined;
|
||||||
|
|
||||||
|
if (inputDevices) {
|
||||||
|
areaIds = inputDevices
|
||||||
|
.filter((device) => device.area_id)
|
||||||
|
.map((device) => device.area_id!);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (inputEntities) {
|
||||||
|
areaIds = (areaIds ?? []).concat(
|
||||||
|
inputEntities
|
||||||
|
.filter((entity) => entity.area_id)
|
||||||
|
.map((entity) => entity.area_id!)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (areaIds) {
|
||||||
|
outputAreas = outputAreas.filter((area) =>
|
||||||
|
areaIds!.includes(area.area_id)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (excludeAreas) {
|
||||||
|
outputAreas = outputAreas.filter(
|
||||||
|
(area) => !excludeAreas!.includes(area.area_id)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (excludeFloors) {
|
||||||
|
outputAreas = outputAreas.filter(
|
||||||
|
(area) => !area.floor_id || !excludeFloors!.includes(area.floor_id)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!outputAreas.length) {
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
id: "no_areas",
|
||||||
|
type: "area",
|
||||||
|
name: this.hass.localize("ui.components.area-picker.no_match"),
|
||||||
|
icon: null,
|
||||||
|
strings: [],
|
||||||
|
level: null,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
const floorAreaLookup = getFloorAreaLookup(outputAreas);
|
||||||
|
const unassisgnedAreas = Object.values(outputAreas).filter(
|
||||||
|
(area) => !area.floor_id || !floorAreaLookup[area.floor_id]
|
||||||
|
);
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const floorAreaEntries: Array<
|
||||||
|
[FloorRegistryEntry | undefined, AreaRegistryEntry[]]
|
||||||
|
> = Object.entries(floorAreaLookup)
|
||||||
|
.map(([floorId, floorAreas]) => {
|
||||||
|
const floor = floors.find((fl) => fl.floor_id === floorId)!;
|
||||||
|
return [floor, floorAreas] as const;
|
||||||
|
})
|
||||||
|
.sort(([floorA], [floorB]) => {
|
||||||
|
if (floorA.level !== floorB.level) {
|
||||||
|
return (floorA.level ?? 0) - (floorB.level ?? 0);
|
||||||
|
}
|
||||||
|
return stringCompare(floorA.name, floorB.name);
|
||||||
|
});
|
||||||
|
|
||||||
|
const output: FloorAreaEntry[] = [];
|
||||||
|
|
||||||
|
floorAreaEntries.forEach(([floor, floorAreas]) => {
|
||||||
|
if (floor) {
|
||||||
|
output.push({
|
||||||
|
id: floor.floor_id,
|
||||||
|
type: "floor",
|
||||||
|
name: floor.name,
|
||||||
|
icon: floor.icon,
|
||||||
|
strings: [floor.floor_id, ...floor.aliases, floor.name],
|
||||||
|
level: floor.level,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
output.push(
|
||||||
|
...floorAreas.map((area, index, array) => ({
|
||||||
|
id: area.area_id,
|
||||||
|
type: "area" as const,
|
||||||
|
name: area.name,
|
||||||
|
icon: area.icon,
|
||||||
|
strings: [area.area_id, ...area.aliases, area.name],
|
||||||
|
hasFloor: true,
|
||||||
|
level: null,
|
||||||
|
lastArea: index === array.length - 1,
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!output.length && !unassisgnedAreas.length) {
|
||||||
|
output.push({
|
||||||
|
id: "no_areas",
|
||||||
|
type: "area",
|
||||||
|
name: this.hass.localize(
|
||||||
|
"ui.components.area-picker.unassigned_areas"
|
||||||
|
),
|
||||||
|
icon: null,
|
||||||
|
strings: [],
|
||||||
|
level: null,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
output.push(
|
||||||
|
...unassisgnedAreas.map((area) => ({
|
||||||
|
id: area.area_id,
|
||||||
|
type: "area" as const,
|
||||||
|
name: area.name,
|
||||||
|
icon: area.icon,
|
||||||
|
strings: [area.area_id, ...area.aliases, area.name],
|
||||||
|
level: null,
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
protected updated(changedProps: PropertyValues) {
|
||||||
|
if (
|
||||||
|
(!this._init && this.hass && this._floors) ||
|
||||||
|
(this._init && changedProps.has("_opened") && this._opened)
|
||||||
|
) {
|
||||||
|
this._init = true;
|
||||||
|
const areas = this._getAreas(
|
||||||
|
this._floors!,
|
||||||
|
Object.values(this.hass.areas),
|
||||||
|
Object.values(this.hass.devices),
|
||||||
|
Object.values(this.hass.entities),
|
||||||
|
this.includeDomains,
|
||||||
|
this.excludeDomains,
|
||||||
|
this.includeDeviceClasses,
|
||||||
|
this.deviceFilter,
|
||||||
|
this.entityFilter,
|
||||||
|
this.excludeAreas,
|
||||||
|
this.excludeFloors
|
||||||
|
);
|
||||||
|
this.comboBox.items = areas;
|
||||||
|
this.comboBox.filteredItems = areas;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<ha-combo-box
|
||||||
|
.hass=${this.hass}
|
||||||
|
.helper=${this.helper}
|
||||||
|
item-value-path="id"
|
||||||
|
item-id-path="id"
|
||||||
|
item-label-path="name"
|
||||||
|
.value=${this._value}
|
||||||
|
.disabled=${this.disabled}
|
||||||
|
.required=${this.required}
|
||||||
|
.label=${this.label === undefined && this.hass
|
||||||
|
? this.hass.localize("ui.components.area-picker.area")
|
||||||
|
: this.label}
|
||||||
|
.placeholder=${this.placeholder
|
||||||
|
? this.hass.areas[this.placeholder]?.name
|
||||||
|
: undefined}
|
||||||
|
.renderer=${this._rowRenderer}
|
||||||
|
@filter-changed=${this._filterChanged}
|
||||||
|
@opened-changed=${this._openedChanged}
|
||||||
|
@value-changed=${this._areaChanged}
|
||||||
|
>
|
||||||
|
</ha-combo-box>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _filterChanged(ev: CustomEvent): void {
|
||||||
|
const target = ev.target as HaComboBox;
|
||||||
|
const filterString = ev.detail.value;
|
||||||
|
if (!filterString) {
|
||||||
|
this.comboBox.filteredItems = this.comboBox.items;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const filteredItems = fuzzyFilterSort<ScorableAreaFloorEntry>(
|
||||||
|
filterString,
|
||||||
|
target.items || []
|
||||||
|
);
|
||||||
|
|
||||||
|
this.comboBox.filteredItems = filteredItems;
|
||||||
|
}
|
||||||
|
|
||||||
|
private get _value() {
|
||||||
|
return this.value || "";
|
||||||
|
}
|
||||||
|
|
||||||
|
private _openedChanged(ev: ValueChangedEvent<boolean>) {
|
||||||
|
this._opened = ev.detail.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _areaChanged(ev: ValueChangedEvent<string>) {
|
||||||
|
ev.stopPropagation();
|
||||||
|
const newValue = ev.detail.value;
|
||||||
|
|
||||||
|
if (newValue === "no_areas") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const selected = this.comboBox.selectedItem;
|
||||||
|
|
||||||
|
fireEvent(this, "value-changed", {
|
||||||
|
value: {
|
||||||
|
id: selected.id,
|
||||||
|
type: selected.type,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-area-floor-picker": HaAreaFloorPicker;
|
||||||
|
}
|
||||||
|
}
|
@@ -1,14 +1,15 @@
|
|||||||
|
import { mdiTextureBox } from "@mdi/js";
|
||||||
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { html, LitElement, nothing, PropertyValues, TemplateResult } from "lit";
|
import { LitElement, PropertyValues, TemplateResult, html } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import { classMap } from "lit/directives/class-map";
|
import { classMap } from "lit/directives/class-map";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
import { computeDomain } from "../common/entity/compute_domain";
|
import { computeDomain } from "../common/entity/compute_domain";
|
||||||
import {
|
import {
|
||||||
fuzzyFilterSort,
|
|
||||||
ScorableTextItem,
|
ScorableTextItem,
|
||||||
|
fuzzyFilterSort,
|
||||||
} from "../common/string/filter/sequence-matching";
|
} from "../common/string/filter/sequence-matching";
|
||||||
import {
|
import {
|
||||||
AreaRegistryEntry,
|
AreaRegistryEntry,
|
||||||
@@ -20,10 +21,8 @@ import {
|
|||||||
getDeviceEntityDisplayLookup,
|
getDeviceEntityDisplayLookup,
|
||||||
} from "../data/device_registry";
|
} from "../data/device_registry";
|
||||||
import { EntityRegistryDisplayEntry } from "../data/entity_registry";
|
import { EntityRegistryDisplayEntry } from "../data/entity_registry";
|
||||||
import {
|
import { showAlertDialog } from "../dialogs/generic/show-dialog-box";
|
||||||
showAlertDialog,
|
import { showAreaRegistryDetailDialog } from "../panels/config/areas/show-dialog-area-registry-detail";
|
||||||
showPromptDialog,
|
|
||||||
} from "../dialogs/generic/show-dialog-box";
|
|
||||||
import { HomeAssistant, ValueChangedEvent } from "../types";
|
import { HomeAssistant, ValueChangedEvent } from "../types";
|
||||||
import type { HaDevicePickerDeviceFilterFunc } from "./device/ha-device-picker";
|
import type { HaDevicePickerDeviceFilterFunc } from "./device/ha-device-picker";
|
||||||
import "./ha-combo-box";
|
import "./ha-combo-box";
|
||||||
@@ -37,14 +36,18 @@ type ScorableAreaRegistryEntry = ScorableTextItem & AreaRegistryEntry;
|
|||||||
const rowRenderer: ComboBoxLitRenderer<AreaRegistryEntry> = (item) =>
|
const rowRenderer: ComboBoxLitRenderer<AreaRegistryEntry> = (item) =>
|
||||||
html`<ha-list-item
|
html`<ha-list-item
|
||||||
graphic="icon"
|
graphic="icon"
|
||||||
class=${classMap({ "add-new": item.area_id === "add_new" })}
|
class=${classMap({ "add-new": item.area_id === ADD_NEW_ID })}
|
||||||
>
|
>
|
||||||
${item.icon
|
${item.icon
|
||||||
? html`<ha-icon slot="graphic" .icon=${item.icon}></ha-icon>`
|
? html`<ha-icon slot="graphic" .icon=${item.icon}></ha-icon>`
|
||||||
: nothing}
|
: html`<ha-svg-icon slot="graphic" .path=${mdiTextureBox}></ha-svg-icon>`}
|
||||||
${item.name}
|
${item.name}
|
||||||
</ha-list-item>`;
|
</ha-list-item>`;
|
||||||
|
|
||||||
|
const ADD_NEW_ID = "___ADD_NEW___";
|
||||||
|
const NO_ITEMS_ID = "___NO_ITEMS___";
|
||||||
|
const ADD_NEW_SUGGESTION_ID = "___ADD_NEW_SUGGESTION___";
|
||||||
|
|
||||||
@customElement("ha-area-picker")
|
@customElement("ha-area-picker")
|
||||||
export class HaAreaPicker extends LitElement {
|
export class HaAreaPicker extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
@@ -133,18 +136,6 @@ export class HaAreaPicker extends LitElement {
|
|||||||
noAdd: this["noAdd"],
|
noAdd: this["noAdd"],
|
||||||
excludeAreas: this["excludeAreas"]
|
excludeAreas: this["excludeAreas"]
|
||||||
): AreaRegistryEntry[] => {
|
): AreaRegistryEntry[] => {
|
||||||
if (!areas.length) {
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
area_id: "no_areas",
|
|
||||||
name: this.hass.localize("ui.components.area-picker.no_areas"),
|
|
||||||
picture: null,
|
|
||||||
icon: null,
|
|
||||||
aliases: [],
|
|
||||||
},
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
let deviceEntityLookup: DeviceEntityDisplayLookup = {};
|
let deviceEntityLookup: DeviceEntityDisplayLookup = {};
|
||||||
let inputDevices: DeviceRegistryEntry[] | undefined;
|
let inputDevices: DeviceRegistryEntry[] | undefined;
|
||||||
let inputEntities: EntityRegistryDisplayEntry[] | undefined;
|
let inputEntities: EntityRegistryDisplayEntry[] | undefined;
|
||||||
@@ -281,11 +272,13 @@ export class HaAreaPicker extends LitElement {
|
|||||||
if (!outputAreas.length) {
|
if (!outputAreas.length) {
|
||||||
outputAreas = [
|
outputAreas = [
|
||||||
{
|
{
|
||||||
area_id: "no_areas",
|
area_id: NO_ITEMS_ID,
|
||||||
name: this.hass.localize("ui.components.area-picker.no_match"),
|
floor_id: null,
|
||||||
|
name: this.hass.localize("ui.components.area-picker.no_areas"),
|
||||||
picture: null,
|
picture: null,
|
||||||
icon: null,
|
icon: null,
|
||||||
aliases: [],
|
aliases: [],
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -295,11 +288,13 @@ export class HaAreaPicker extends LitElement {
|
|||||||
: [
|
: [
|
||||||
...outputAreas,
|
...outputAreas,
|
||||||
{
|
{
|
||||||
area_id: "add_new",
|
area_id: ADD_NEW_ID,
|
||||||
|
floor_id: null,
|
||||||
name: this.hass.localize("ui.components.area-picker.add_new"),
|
name: this.hass.localize("ui.components.area-picker.add_new"),
|
||||||
picture: null,
|
picture: null,
|
||||||
icon: "mdi:plus",
|
icon: "mdi:plus",
|
||||||
aliases: [],
|
aliases: [],
|
||||||
|
labels: [],
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
@@ -367,20 +362,40 @@ export class HaAreaPicker extends LitElement {
|
|||||||
|
|
||||||
const filteredItems = fuzzyFilterSort<ScorableAreaRegistryEntry>(
|
const filteredItems = fuzzyFilterSort<ScorableAreaRegistryEntry>(
|
||||||
filterString,
|
filterString,
|
||||||
target.items || []
|
target.items?.filter(
|
||||||
|
(item) => ![NO_ITEMS_ID, ADD_NEW_ID].includes(item.label_id)
|
||||||
|
) || []
|
||||||
);
|
);
|
||||||
if (!this.noAdd && filteredItems?.length === 0) {
|
if (filteredItems.length === 0) {
|
||||||
this._suggestion = filterString;
|
if (!this.noAdd) {
|
||||||
this.comboBox.filteredItems = [
|
this.comboBox.filteredItems = [
|
||||||
{
|
{
|
||||||
area_id: "add_new_suggestion",
|
area_id: NO_ITEMS_ID,
|
||||||
name: this.hass.localize(
|
floor_id: null,
|
||||||
"ui.components.area-picker.add_new_sugestion",
|
name: this.hass.localize("ui.components.area-picker.no_match"),
|
||||||
{ name: this._suggestion }
|
icon: null,
|
||||||
),
|
picture: null,
|
||||||
picture: null,
|
labels: [],
|
||||||
},
|
aliases: [],
|
||||||
];
|
},
|
||||||
|
] as AreaRegistryEntry[];
|
||||||
|
} else {
|
||||||
|
this._suggestion = filterString;
|
||||||
|
this.comboBox.filteredItems = [
|
||||||
|
{
|
||||||
|
area_id: ADD_NEW_SUGGESTION_ID,
|
||||||
|
floor_id: null,
|
||||||
|
name: this.hass.localize(
|
||||||
|
"ui.components.area-picker.add_new_sugestion",
|
||||||
|
{ name: this._suggestion }
|
||||||
|
),
|
||||||
|
icon: "mdi:plus",
|
||||||
|
picture: null,
|
||||||
|
labels: [],
|
||||||
|
aliases: [],
|
||||||
|
},
|
||||||
|
] as AreaRegistryEntry[];
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
this.comboBox.filteredItems = filteredItems;
|
this.comboBox.filteredItems = filteredItems;
|
||||||
}
|
}
|
||||||
@@ -398,11 +413,13 @@ export class HaAreaPicker extends LitElement {
|
|||||||
ev.stopPropagation();
|
ev.stopPropagation();
|
||||||
let newValue = ev.detail.value;
|
let newValue = ev.detail.value;
|
||||||
|
|
||||||
if (newValue === "no_areas") {
|
if (newValue === NO_ITEMS_ID) {
|
||||||
newValue = "";
|
newValue = "";
|
||||||
|
this.comboBox.setInputValue("");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!["add_new_suggestion", "add_new"].includes(newValue)) {
|
if (![ADD_NEW_SUGGESTION_ID, ADD_NEW_ID].includes(newValue)) {
|
||||||
if (newValue !== this._value) {
|
if (newValue !== this._value) {
|
||||||
this._setValue(newValue);
|
this._setValue(newValue);
|
||||||
}
|
}
|
||||||
@@ -410,25 +427,14 @@ export class HaAreaPicker extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
(ev.target as any).value = this._value;
|
(ev.target as any).value = this._value;
|
||||||
showPromptDialog(this, {
|
|
||||||
title: this.hass.localize("ui.components.area-picker.add_dialog.title"),
|
this.hass.loadFragmentTranslation("config");
|
||||||
text: this.hass.localize("ui.components.area-picker.add_dialog.text"),
|
|
||||||
confirmText: this.hass.localize(
|
showAreaRegistryDetailDialog(this, {
|
||||||
"ui.components.area-picker.add_dialog.add"
|
suggestedName: newValue === ADD_NEW_SUGGESTION_ID ? this._suggestion : "",
|
||||||
),
|
createEntry: async (values) => {
|
||||||
inputLabel: this.hass.localize(
|
|
||||||
"ui.components.area-picker.add_dialog.name"
|
|
||||||
),
|
|
||||||
defaultValue:
|
|
||||||
newValue === "add_new_suggestion" ? this._suggestion : undefined,
|
|
||||||
confirm: async (name) => {
|
|
||||||
if (!name) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try {
|
try {
|
||||||
const area = await createAreaRegistryEntry(this.hass, {
|
const area = await createAreaRegistryEntry(this.hass, values);
|
||||||
name,
|
|
||||||
});
|
|
||||||
const areas = [...Object.values(this.hass.areas), area];
|
const areas = [...Object.values(this.hass.areas), area];
|
||||||
this.comboBox.filteredItems = this._getAreas(
|
this.comboBox.filteredItems = this._getAreas(
|
||||||
areas,
|
areas,
|
||||||
@@ -448,18 +454,16 @@ export class HaAreaPicker extends LitElement {
|
|||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: this.hass.localize(
|
title: this.hass.localize(
|
||||||
"ui.components.area-picker.add_dialog.failed_create_area"
|
"ui.components.area-picker.failed_create_area"
|
||||||
),
|
),
|
||||||
text: err.message,
|
text: err.message,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
cancel: () => {
|
|
||||||
this._setValue(undefined);
|
|
||||||
this._suggestion = undefined;
|
|
||||||
this.comboBox.setInputValue("");
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this._suggestion = undefined;
|
||||||
|
this.comboBox.setInputValue("");
|
||||||
}
|
}
|
||||||
|
|
||||||
private _setValue(value?: string) {
|
private _setValue(value?: string) {
|
||||||
|
89
src/components/ha-button-menu-new.ts
Normal file
89
src/components/ha-button-menu-new.ts
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
import { Button } from "@material/mwc-button";
|
||||||
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement, property, query } from "lit/decorators";
|
||||||
|
import { FOCUS_TARGET } from "../dialogs/make-dialog-manager";
|
||||||
|
import type { HaIconButton } from "./ha-icon-button";
|
||||||
|
import "./ha-menu";
|
||||||
|
import type { HaMenu } from "./ha-menu";
|
||||||
|
|
||||||
|
@customElement("ha-button-menu-new")
|
||||||
|
export class HaButtonMenuNew extends LitElement {
|
||||||
|
protected readonly [FOCUS_TARGET];
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
|
@property() public positioning?: "fixed" | "absolute" | "popover";
|
||||||
|
|
||||||
|
@property({ type: Boolean, attribute: "has-overflow" }) public hasOverflow =
|
||||||
|
false;
|
||||||
|
|
||||||
|
@query("ha-menu", true) private _menu!: HaMenu;
|
||||||
|
|
||||||
|
public get items() {
|
||||||
|
return this._menu.items;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override focus() {
|
||||||
|
if (this._menu.open) {
|
||||||
|
this._menu.focus();
|
||||||
|
} else {
|
||||||
|
this._triggerButton?.focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<div @click=${this._handleClick}>
|
||||||
|
<slot name="trigger" @slotchange=${this._setTriggerAria}></slot>
|
||||||
|
</div>
|
||||||
|
<ha-menu
|
||||||
|
.positioning=${this.positioning}
|
||||||
|
.hasOverflow=${this.hasOverflow}
|
||||||
|
>
|
||||||
|
<slot></slot>
|
||||||
|
</ha-menu>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleClick(): void {
|
||||||
|
if (this.disabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this._menu.anchorElement = this;
|
||||||
|
if (this._menu.open) {
|
||||||
|
this._menu.close();
|
||||||
|
} else {
|
||||||
|
this._menu.show();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private get _triggerButton() {
|
||||||
|
return this.querySelector(
|
||||||
|
'ha-icon-button[slot="trigger"], mwc-button[slot="trigger"], ha-assist-chip[slot="trigger"]'
|
||||||
|
) as HaIconButton | Button | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _setTriggerAria() {
|
||||||
|
if (this._triggerButton) {
|
||||||
|
this._triggerButton.ariaHasPopup = "menu";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResultGroup {
|
||||||
|
return css`
|
||||||
|
:host {
|
||||||
|
display: inline-block;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
::slotted([disabled]) {
|
||||||
|
color: var(--disabled-text-color);
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-button-menu-new": HaButtonMenuNew;
|
||||||
|
}
|
||||||
|
}
|
@@ -1,221 +0,0 @@
|
|||||||
import type { Corner } from "@material/mwc-menu";
|
|
||||||
import "@material/mwc-menu/mwc-menu-surface";
|
|
||||||
import { mdiFilterVariant } from "@mdi/js";
|
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
|
||||||
import { customElement, property, state } from "lit/decorators";
|
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
|
||||||
import { stopPropagation } from "../common/dom/stop_propagation";
|
|
||||||
import { computeStateName } from "../common/entity/compute_state_name";
|
|
||||||
import { computeDeviceName } from "../data/device_registry";
|
|
||||||
import { findRelated, RelatedResult } from "../data/search";
|
|
||||||
import type { HomeAssistant } from "../types";
|
|
||||||
import "./device/ha-device-picker";
|
|
||||||
import "./entity/ha-entity-picker";
|
|
||||||
import "./ha-area-picker";
|
|
||||||
import "./ha-icon-button";
|
|
||||||
|
|
||||||
declare global {
|
|
||||||
// for fire event
|
|
||||||
interface HASSDomEvents {
|
|
||||||
"related-changed": {
|
|
||||||
value?: FilterValue;
|
|
||||||
items?: RelatedResult;
|
|
||||||
filter?: string;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FilterValue {
|
|
||||||
area?: string;
|
|
||||||
device?: string;
|
|
||||||
entity?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
@customElement("ha-button-related-filter-menu")
|
|
||||||
export class HaRelatedFilterButtonMenu extends LitElement {
|
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
|
||||||
|
|
||||||
@property() public corner: Corner = "BOTTOM_START";
|
|
||||||
|
|
||||||
@property({ type: Boolean, reflect: true }) public narrow = false;
|
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public value?: FilterValue;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Show no entities of these domains.
|
|
||||||
* @type {Array}
|
|
||||||
* @attr exclude-domains
|
|
||||||
*/
|
|
||||||
@property({ type: Array, attribute: "exclude-domains" })
|
|
||||||
public excludeDomains?: string[];
|
|
||||||
|
|
||||||
@state() private _open = false;
|
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
|
||||||
return html`
|
|
||||||
<ha-icon-button
|
|
||||||
@click=${this._handleClick}
|
|
||||||
.label=${this.hass.localize("ui.components.related-filter-menu.filter")}
|
|
||||||
.path=${mdiFilterVariant}
|
|
||||||
></ha-icon-button>
|
|
||||||
<mwc-menu-surface
|
|
||||||
.open=${this._open}
|
|
||||||
.anchor=${this}
|
|
||||||
.fullwidth=${this.narrow}
|
|
||||||
.corner=${this.corner}
|
|
||||||
@closed=${this._onClosed}
|
|
||||||
@input=${stopPropagation}
|
|
||||||
>
|
|
||||||
<ha-area-picker
|
|
||||||
.label=${this.hass.localize(
|
|
||||||
"ui.components.related-filter-menu.filter_by_area"
|
|
||||||
)}
|
|
||||||
.hass=${this.hass}
|
|
||||||
.value=${this.value?.area}
|
|
||||||
no-add
|
|
||||||
@value-changed=${this._areaPicked}
|
|
||||||
@click=${this._preventDefault}
|
|
||||||
></ha-area-picker>
|
|
||||||
<ha-device-picker
|
|
||||||
.label=${this.hass.localize(
|
|
||||||
"ui.components.related-filter-menu.filter_by_device"
|
|
||||||
)}
|
|
||||||
.hass=${this.hass}
|
|
||||||
.value=${this.value?.device}
|
|
||||||
@value-changed=${this._devicePicked}
|
|
||||||
@click=${this._preventDefault}
|
|
||||||
></ha-device-picker>
|
|
||||||
<ha-entity-picker
|
|
||||||
.label=${this.hass.localize(
|
|
||||||
"ui.components.related-filter-menu.filter_by_entity"
|
|
||||||
)}
|
|
||||||
.hass=${this.hass}
|
|
||||||
.value=${this.value?.entity}
|
|
||||||
.excludeDomains=${this.excludeDomains}
|
|
||||||
@value-changed=${this._entityPicked}
|
|
||||||
@click=${this._preventDefault}
|
|
||||||
></ha-entity-picker>
|
|
||||||
</mwc-menu-surface>
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
|
|
||||||
private _handleClick(): void {
|
|
||||||
if (this.disabled) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
this._open = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private _onClosed(ev): void {
|
|
||||||
ev.stopPropagation();
|
|
||||||
this._open = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private _preventDefault(ev) {
|
|
||||||
ev.preventDefault();
|
|
||||||
}
|
|
||||||
|
|
||||||
private async _entityPicked(ev: CustomEvent) {
|
|
||||||
ev.stopPropagation();
|
|
||||||
const entityId = ev.detail.value;
|
|
||||||
if (!entityId) {
|
|
||||||
fireEvent(this, "related-changed", { value: undefined });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const filter = this.hass.localize(
|
|
||||||
"ui.components.related-filter-menu.filtered_by_entity",
|
|
||||||
{
|
|
||||||
entity_name: computeStateName(
|
|
||||||
(ev.currentTarget as any).comboBox.selectedItem
|
|
||||||
),
|
|
||||||
}
|
|
||||||
);
|
|
||||||
const items = await findRelated(this.hass, "entity", entityId);
|
|
||||||
fireEvent(this, "related-changed", {
|
|
||||||
value: { entity: entityId },
|
|
||||||
filter,
|
|
||||||
items,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private async _devicePicked(ev: CustomEvent) {
|
|
||||||
ev.stopPropagation();
|
|
||||||
const deviceId = ev.detail.value;
|
|
||||||
if (!deviceId) {
|
|
||||||
fireEvent(this, "related-changed", { value: undefined });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const filter = this.hass.localize(
|
|
||||||
"ui.components.related-filter-menu.filtered_by_device",
|
|
||||||
{
|
|
||||||
device_name: computeDeviceName(
|
|
||||||
(ev.currentTarget as any).comboBox.selectedItem,
|
|
||||||
this.hass
|
|
||||||
),
|
|
||||||
}
|
|
||||||
);
|
|
||||||
const items = await findRelated(this.hass, "device", deviceId);
|
|
||||||
|
|
||||||
fireEvent(this, "related-changed", {
|
|
||||||
value: { device: deviceId },
|
|
||||||
filter,
|
|
||||||
items,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private async _areaPicked(ev: CustomEvent) {
|
|
||||||
ev.stopPropagation();
|
|
||||||
const areaId = ev.detail.value;
|
|
||||||
if (!areaId) {
|
|
||||||
fireEvent(this, "related-changed", { value: undefined });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const filter = this.hass.localize(
|
|
||||||
"ui.components.related-filter-menu.filtered_by_area",
|
|
||||||
{ area_name: (ev.currentTarget as any).comboBox.selectedItem.name }
|
|
||||||
);
|
|
||||||
const items = await findRelated(this.hass, "area", areaId);
|
|
||||||
fireEvent(this, "related-changed", {
|
|
||||||
value: { area: areaId },
|
|
||||||
filter,
|
|
||||||
items,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
static get styles(): CSSResultGroup {
|
|
||||||
return css`
|
|
||||||
:host {
|
|
||||||
display: inline-block;
|
|
||||||
position: relative;
|
|
||||||
--mdc-menu-min-width: 250px;
|
|
||||||
}
|
|
||||||
ha-area-picker,
|
|
||||||
ha-device-picker,
|
|
||||||
ha-entity-picker {
|
|
||||||
display: block;
|
|
||||||
width: 300px;
|
|
||||||
padding: 4px 16px;
|
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
|
||||||
ha-area-picker {
|
|
||||||
padding-top: 16px;
|
|
||||||
}
|
|
||||||
ha-entity-picker {
|
|
||||||
padding-bottom: 16px;
|
|
||||||
}
|
|
||||||
:host([narrow]) ha-area-picker,
|
|
||||||
:host([narrow]) ha-device-picker,
|
|
||||||
:host([narrow]) ha-entity-picker {
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
declare global {
|
|
||||||
interface HTMLElementTagNameMap {
|
|
||||||
"ha-button-related-filter-menu": HaRelatedFilterButtonMenu;
|
|
||||||
}
|
|
||||||
}
|
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user