mirror of
https://github.com/home-assistant/frontend.git
synced 2025-10-24 11:09:48 +00:00
Compare commits
600 Commits
20210301.0
...
blocking-u
Author | SHA1 | Date | |
---|---|---|---|
![]() |
09f4922ad3 | ||
![]() |
36831d26e4 | ||
![]() |
5829660894 | ||
![]() |
4e3fbc1169 | ||
![]() |
38640c99e3 | ||
![]() |
d6df8bddea | ||
![]() |
ddfc4bd98e | ||
![]() |
3d6674325c | ||
![]() |
194829f5b1 | ||
![]() |
11a77253f4 | ||
![]() |
67be2343f8 | ||
![]() |
e9b1b3d853 | ||
![]() |
8a33d174d7 | ||
![]() |
226d6216b7 | ||
![]() |
1925bb01be | ||
![]() |
82a4806e01 | ||
![]() |
ce419fae7b | ||
![]() |
c68b76e2da | ||
![]() |
342020b420 | ||
![]() |
1e6e99e3c7 | ||
![]() |
2e9aafc377 | ||
![]() |
299c863f49 | ||
![]() |
c2792a28ba | ||
![]() |
635a027a8e | ||
![]() |
a45b8ca8e7 | ||
![]() |
1e6e945a07 | ||
![]() |
f71157c24d | ||
![]() |
e87a2b36cf | ||
![]() |
5418474f64 | ||
![]() |
8836ba6ceb | ||
![]() |
509c5b497a | ||
![]() |
e00bcc9f48 | ||
![]() |
bdef9fd040 | ||
![]() |
c956491ec5 | ||
![]() |
68bc549d6a | ||
![]() |
9c64eafc21 | ||
![]() |
b05e86d442 | ||
![]() |
fe5f9576c6 | ||
![]() |
1b282b65b7 | ||
![]() |
e49664bad3 | ||
![]() |
2a30b55a43 | ||
![]() |
9d0b20adce | ||
![]() |
acd5e1c081 | ||
![]() |
cc1c5e45b2 | ||
![]() |
038199c447 | ||
![]() |
8a1eab7ceb | ||
![]() |
bc5bd35448 | ||
![]() |
1795fd56b7 | ||
![]() |
4a7c33edad | ||
![]() |
797f60d725 | ||
![]() |
2427d68aa1 | ||
![]() |
00c6b0f8ed | ||
![]() |
7b8d4ab3d6 | ||
![]() |
07a1a805f6 | ||
![]() |
d8bab6aba9 | ||
![]() |
a930e2dc75 | ||
![]() |
2eb35668fa | ||
![]() |
07f4e5ac5c | ||
![]() |
db82a90414 | ||
![]() |
51a693badf | ||
![]() |
2aa8f5b352 | ||
![]() |
93b3b8f985 | ||
![]() |
92c8bd80b5 | ||
![]() |
528af0157d | ||
![]() |
10a77b6278 | ||
![]() |
03bbf6a582 | ||
![]() |
63fcb649d2 | ||
![]() |
4f60a92b92 | ||
![]() |
0419c1a41f | ||
![]() |
2d5ae78521 | ||
![]() |
959134df02 | ||
![]() |
a9f9fc4ce2 | ||
![]() |
cfb370a3c8 | ||
![]() |
353435c8d5 | ||
![]() |
c8c85d096b | ||
![]() |
19c9c8f227 | ||
![]() |
6ea2a29eea | ||
![]() |
59f3f819a6 | ||
![]() |
93e8f52880 | ||
![]() |
02810efcc4 | ||
![]() |
4b9be7ce16 | ||
![]() |
f3ec09e480 | ||
![]() |
8291a84e3e | ||
![]() |
b0e1f0f73a | ||
![]() |
a66b966e7d | ||
![]() |
5f56040c64 | ||
![]() |
eaccd22267 | ||
![]() |
27845a7345 | ||
![]() |
f7ef8180e4 | ||
![]() |
5958eb9a55 | ||
![]() |
3ef2912b60 | ||
![]() |
fa9c6a765a | ||
![]() |
c4a8899780 | ||
![]() |
3cc4628d03 | ||
![]() |
b6c5223221 | ||
![]() |
cbd6d4251c | ||
![]() |
fdcbb5b432 | ||
![]() |
de09e31815 | ||
![]() |
f55e911313 | ||
![]() |
465a91dbf3 | ||
![]() |
835a7833ae | ||
![]() |
179717d40c | ||
![]() |
3d4d789f7f | ||
![]() |
d97fb19f05 | ||
![]() |
0dd3757df2 | ||
![]() |
c32a4546f3 | ||
![]() |
1bb025ccd0 | ||
![]() |
2b8033a97f | ||
![]() |
21a3a8c594 | ||
![]() |
1026e90296 | ||
![]() |
0eca602e61 | ||
![]() |
7f75ca81f1 | ||
![]() |
8af05e2726 | ||
![]() |
0a478ee1da | ||
![]() |
a4bdc5a05f | ||
![]() |
d425767dae | ||
![]() |
c78382c119 | ||
![]() |
ee15ddfbc3 | ||
![]() |
0af14eb77e | ||
![]() |
583cc4bc8a | ||
![]() |
2ee92f48e6 | ||
![]() |
d05e02ab3e | ||
![]() |
abb9f8e233 | ||
![]() |
f873ef9b59 | ||
![]() |
1255b56522 | ||
![]() |
fd9bb4d8cc | ||
![]() |
9328576b55 | ||
![]() |
70a1edd1dd | ||
![]() |
87e4c209f4 | ||
![]() |
3d0a5642cc | ||
![]() |
e211d812ad | ||
![]() |
0dcf673b87 | ||
![]() |
cb14e1f20c | ||
![]() |
52087c0e30 | ||
![]() |
1b9286db76 | ||
![]() |
bc92c0b052 | ||
![]() |
245bb639f2 | ||
![]() |
8d81ed58c8 | ||
![]() |
7890ca85a8 | ||
![]() |
07bab7b264 | ||
![]() |
5730c14dc1 | ||
![]() |
f8e8b5ad18 | ||
![]() |
fd2728c02c | ||
![]() |
7e2bf920e1 | ||
![]() |
1f65328f2d | ||
![]() |
4f731baa00 | ||
![]() |
5abb3dd8c1 | ||
![]() |
0a672c55c5 | ||
![]() |
a6b2299c74 | ||
![]() |
37cc6709d4 | ||
![]() |
f4ffbe67e2 | ||
![]() |
9f32d72a41 | ||
![]() |
64a117d8ac | ||
![]() |
ebf0bdc840 | ||
![]() |
cc0a120bf6 | ||
![]() |
fe2fe7468f | ||
![]() |
b12a10ccb5 | ||
![]() |
2ad2a4b198 | ||
![]() |
6a62f05657 | ||
![]() |
4910f60ec4 | ||
![]() |
d35168e88f | ||
![]() |
01b3d2aca9 | ||
![]() |
29e8d1cff0 | ||
![]() |
4e1d10cc08 | ||
![]() |
3575d94ca1 | ||
![]() |
d91546b532 | ||
![]() |
9f554f4917 | ||
![]() |
d4720a9244 | ||
![]() |
5c466712db | ||
![]() |
6dc7e852ae | ||
![]() |
785f614bd9 | ||
![]() |
0a8e27249d | ||
![]() |
15ee87ee67 | ||
![]() |
12612a16df | ||
![]() |
4f449e2600 | ||
![]() |
7f49f039fd | ||
![]() |
88dc65bc4e | ||
![]() |
6edebe18ad | ||
![]() |
38b3a9205d | ||
![]() |
4b796b4929 | ||
![]() |
83cabcac28 | ||
![]() |
d308c5d9b9 | ||
![]() |
9f032a61a9 | ||
![]() |
0f58214ba1 | ||
![]() |
c48a60cce6 | ||
![]() |
cd3ffceeff | ||
![]() |
9be4a00169 | ||
![]() |
a9c7a39a47 | ||
![]() |
abcdd60a21 | ||
![]() |
a94f85a100 | ||
![]() |
9755bf723f | ||
![]() |
a71ebcf47e | ||
![]() |
72695631cd | ||
![]() |
2af211b543 | ||
![]() |
6e5e2625d6 | ||
![]() |
23c1c2f5eb | ||
![]() |
da85ee5d01 | ||
![]() |
d408e8653c | ||
![]() |
cc76ccc3c9 | ||
![]() |
105a00d3e4 | ||
![]() |
2c08cba8cc | ||
![]() |
344b11a204 | ||
![]() |
1ff5bf0fd5 | ||
![]() |
c29cf7f77c | ||
![]() |
193cb46d60 | ||
![]() |
9dc864d486 | ||
![]() |
cee166839a | ||
![]() |
1a60a3c728 | ||
![]() |
5d946778cb | ||
![]() |
ac5f85820f | ||
![]() |
716e100a28 | ||
![]() |
7b8cb16c12 | ||
![]() |
00d46424a3 | ||
![]() |
2a5f940744 | ||
![]() |
13cc016b36 | ||
![]() |
a8d49c27c8 | ||
![]() |
a8522e91b5 | ||
![]() |
5754f4463d | ||
![]() |
d4118ade0f | ||
![]() |
6d80f15a98 | ||
![]() |
f8aa472409 | ||
![]() |
df22fd00ca | ||
![]() |
ce2743a982 | ||
![]() |
92b32458ad | ||
![]() |
d57e8a45d3 | ||
![]() |
551d3ffdf3 | ||
![]() |
7add6eb736 | ||
![]() |
a28616d535 | ||
![]() |
a288fd370f | ||
![]() |
acd335e249 | ||
![]() |
da0bfa1945 | ||
![]() |
3c61d709b5 | ||
![]() |
ffc92a7b63 | ||
![]() |
af0c7b5a50 | ||
![]() |
1904c4d057 | ||
![]() |
542f169b36 | ||
![]() |
65a30bf60c | ||
![]() |
2e51da32f0 | ||
![]() |
0562242043 | ||
![]() |
debcdefc21 | ||
![]() |
0de3f3a332 | ||
![]() |
4fcb4d449e | ||
![]() |
408fe25abd | ||
![]() |
236e5e0b25 | ||
![]() |
ebe0caba83 | ||
![]() |
9d33c0cfaf | ||
![]() |
7962130a0c | ||
![]() |
9690434cac | ||
![]() |
7304544c37 | ||
![]() |
5a3408c242 | ||
![]() |
16996f25af | ||
![]() |
0c12586019 | ||
![]() |
93a1adaa56 | ||
![]() |
83e65e2cc6 | ||
![]() |
36586b798e | ||
![]() |
20c351949f | ||
![]() |
b63bd92d81 | ||
![]() |
9f3bb7f4d6 | ||
![]() |
73bb346c00 | ||
![]() |
33703a3b53 | ||
![]() |
b7a4f97eca | ||
![]() |
dd4efe0f51 | ||
![]() |
7e0522c3b3 | ||
![]() |
e682abfb75 | ||
![]() |
24e202a3d7 | ||
![]() |
ac9a881ab5 | ||
![]() |
4d287a1f83 | ||
![]() |
b8d6b1ebdd | ||
![]() |
8ca1b9320d | ||
![]() |
cba3992d2b | ||
![]() |
96d6e337be | ||
![]() |
959f7ae046 | ||
![]() |
9572a58764 | ||
![]() |
393ae9e5dc | ||
![]() |
63e10314bd | ||
![]() |
b599417a37 | ||
![]() |
899eab4e5c | ||
![]() |
3f21c87a3d | ||
![]() |
c296a60bab | ||
![]() |
5f78f18cb4 | ||
![]() |
0b8d356865 | ||
![]() |
e8d1318a5b | ||
![]() |
07ce07c4a5 | ||
![]() |
a07220f383 | ||
![]() |
f21ed24a49 | ||
![]() |
e3c38b93f4 | ||
![]() |
b398727413 | ||
![]() |
9bc2ab29a1 | ||
![]() |
51f1ff26f1 | ||
![]() |
97d5e6512d | ||
![]() |
b76c67fc9b | ||
![]() |
b96a70cd55 | ||
![]() |
982ab93cdb | ||
![]() |
c7f4e1152d | ||
![]() |
519988326b | ||
![]() |
b518f4b03c | ||
![]() |
5493fdfcb7 | ||
![]() |
179767e9f8 | ||
![]() |
25b3bb1285 | ||
![]() |
841c8ab1f1 | ||
![]() |
1ce17e2847 | ||
![]() |
a09b206b0e | ||
![]() |
bb4617c53b | ||
![]() |
cfd18bfb74 | ||
![]() |
e225d6f546 | ||
![]() |
60fe48d355 | ||
![]() |
2dcd0d2b0a | ||
![]() |
8e11aa9130 | ||
![]() |
f6e223c18d | ||
![]() |
9d29b55bee | ||
![]() |
92aa8580db | ||
![]() |
538028a003 | ||
![]() |
c53575a74f | ||
![]() |
193016a46a | ||
![]() |
aaa50b4d1d | ||
![]() |
a43120320e | ||
![]() |
b8bb0c038d | ||
![]() |
dc79fc2919 | ||
![]() |
30787fef60 | ||
![]() |
445ae156ef | ||
![]() |
62a0cfb0f6 | ||
![]() |
96bc3ef99a | ||
![]() |
1d3b95d24f | ||
![]() |
56fe4b07f3 | ||
![]() |
ea60f7005b | ||
![]() |
9eb59062aa | ||
![]() |
d00927c31f | ||
![]() |
c03017208d | ||
![]() |
73f945458a | ||
![]() |
db12234611 | ||
![]() |
ed1cd4632f | ||
![]() |
17d3755152 | ||
![]() |
d7c0c2ea72 | ||
![]() |
7c823c98ae | ||
![]() |
97508a6f31 | ||
![]() |
2507a41b6e | ||
![]() |
9833accc79 | ||
![]() |
d46123771a | ||
![]() |
87fe84b1ac | ||
![]() |
21140f437e | ||
![]() |
ba9e410393 | ||
![]() |
9b628546c1 | ||
![]() |
d0837fada8 | ||
![]() |
520647d72f | ||
![]() |
51c888845c | ||
![]() |
e4606219bc | ||
![]() |
716335df2c | ||
![]() |
ad4f90c502 | ||
![]() |
a1bdfa7560 | ||
![]() |
587fb2a170 | ||
![]() |
7d801ff84c | ||
![]() |
d69accd9a5 | ||
![]() |
1127750c5e | ||
![]() |
7758bd89c1 | ||
![]() |
de7264327a | ||
![]() |
c3f0932794 | ||
![]() |
367907e037 | ||
![]() |
2d15bd651e | ||
![]() |
4b1d7863f8 | ||
![]() |
e425d768dd | ||
![]() |
34ca807044 | ||
![]() |
9075146b47 | ||
![]() |
26c4591baa | ||
![]() |
2aac8c55e7 | ||
![]() |
8af55efdb3 | ||
![]() |
9d6e07ff96 | ||
![]() |
8f58eee6af | ||
![]() |
8dd3d78f21 | ||
![]() |
48161fd02f | ||
![]() |
b61410826d | ||
![]() |
2f0188b280 | ||
![]() |
3a4fffdb0b | ||
![]() |
6393072e68 | ||
![]() |
109910d18f | ||
![]() |
8874aaabe9 | ||
![]() |
cafbea9c42 | ||
![]() |
4843ee80a7 | ||
![]() |
4511c8f30c | ||
![]() |
4cf1e52ac0 | ||
![]() |
b501b7f47c | ||
![]() |
cc275f9877 | ||
![]() |
7aae55cde7 | ||
![]() |
85eaa219c6 | ||
![]() |
7d5ecb8ba4 | ||
![]() |
1fd142d337 | ||
![]() |
d75c6aecbe | ||
![]() |
dffe0f656d | ||
![]() |
890639436b | ||
![]() |
99f66d7c5d | ||
![]() |
05faa52425 | ||
![]() |
9e1a8b646b | ||
![]() |
8f6ec03446 | ||
![]() |
c56b4fade3 | ||
![]() |
61aaaabcb5 | ||
![]() |
d57cf93580 | ||
![]() |
82ad5c103d | ||
![]() |
a0b5bc5456 | ||
![]() |
c810e541ea | ||
![]() |
05ea3b8187 | ||
![]() |
8301dffb21 | ||
![]() |
01be5243de | ||
![]() |
334196799a | ||
![]() |
c11bbcf442 | ||
![]() |
8e3a7576ea | ||
![]() |
deca6f03ba | ||
![]() |
401064d3c8 | ||
![]() |
b6f59d3c98 | ||
![]() |
1fb3663398 | ||
![]() |
5c1604e959 | ||
![]() |
17b1f3e465 | ||
![]() |
9a68bdeec1 | ||
![]() |
9b947ef734 | ||
![]() |
66432608ed | ||
![]() |
d8153ac8fc | ||
![]() |
27d9f82f7d | ||
![]() |
5b55bcd879 | ||
![]() |
5cfd28881b | ||
![]() |
bc54a42e01 | ||
![]() |
03f9964c59 | ||
![]() |
f159219d2c | ||
![]() |
e714f32737 | ||
![]() |
20858db96d | ||
![]() |
89b82bb778 | ||
![]() |
2c886d739f | ||
![]() |
1ccf4e49bc | ||
![]() |
7d63e3e088 | ||
![]() |
828523f281 | ||
![]() |
afe3831f25 | ||
![]() |
3888c56f1a | ||
![]() |
6f07966ef8 | ||
![]() |
09eafe8abd | ||
![]() |
6719a42e27 | ||
![]() |
4b98a70ee8 | ||
![]() |
db3f5447ca | ||
![]() |
fed63f645d | ||
![]() |
e7315bb570 | ||
![]() |
cd2404f26a | ||
![]() |
b866166425 | ||
![]() |
46580376dd | ||
![]() |
b8bfb44aec | ||
![]() |
a153f572d0 | ||
![]() |
66c30a59e7 | ||
![]() |
10b8efc5cb | ||
![]() |
c65d414b7b | ||
![]() |
7f7d89c745 | ||
![]() |
742028b691 | ||
![]() |
62f685bac2 | ||
![]() |
0b3333e88c | ||
![]() |
c341a99b83 | ||
![]() |
f43c420d59 | ||
![]() |
0393970a80 | ||
![]() |
1865e0661f | ||
![]() |
c07b1194b3 | ||
![]() |
bf802628b9 | ||
![]() |
36020373cd | ||
![]() |
43e73d69de | ||
![]() |
47a3f649d2 | ||
![]() |
5c63f8e52a | ||
![]() |
01c553ef13 | ||
![]() |
f229e4e12a | ||
![]() |
40cf4c8d32 | ||
![]() |
ee38c419de | ||
![]() |
10baa34c18 | ||
![]() |
343b67fa7f | ||
![]() |
6de8b4e35f | ||
![]() |
57e535c2c8 | ||
![]() |
af5b22a265 | ||
![]() |
77972c961b | ||
![]() |
a3efa5676b | ||
![]() |
014dbc2a86 | ||
![]() |
226a2941d6 | ||
![]() |
c269c8fd3f | ||
![]() |
d8fc3c1ebf | ||
![]() |
a5c6ffd1b9 | ||
![]() |
9aaaaae175 | ||
![]() |
7d39b69540 | ||
![]() |
09bad14c3d | ||
![]() |
369c9dc6e2 | ||
![]() |
9676d2cee7 | ||
![]() |
5156c67226 | ||
![]() |
4d48fc3d85 | ||
![]() |
20da329a21 | ||
![]() |
4b664cc142 | ||
![]() |
c9b620fdb2 | ||
![]() |
25c886d401 | ||
![]() |
740805356f | ||
![]() |
ce5fb57577 | ||
![]() |
3e20d2b454 | ||
![]() |
a9e8186491 | ||
![]() |
3bb909b026 | ||
![]() |
b921d91aeb | ||
![]() |
05790954c6 | ||
![]() |
13014c1351 | ||
![]() |
e34c63b830 | ||
![]() |
943100d758 | ||
![]() |
55f40d66f2 | ||
![]() |
593e5ac79c | ||
![]() |
ef31bce5ee | ||
![]() |
3c75eb96f1 | ||
![]() |
f34dfde925 | ||
![]() |
e3b72fe0aa | ||
![]() |
60de74a375 | ||
![]() |
55e58f8d35 | ||
![]() |
a465254418 | ||
![]() |
5d27a138cf | ||
![]() |
22f4b036df | ||
![]() |
03f694922d | ||
![]() |
a841e287e5 | ||
![]() |
5d2afdd825 | ||
![]() |
67240e2339 | ||
![]() |
f84a8eccfa | ||
![]() |
68a058e4f1 | ||
![]() |
d678b42ece | ||
![]() |
2cf63cda08 | ||
![]() |
7bd4eeb0df | ||
![]() |
dc3ee7c779 | ||
![]() |
e8cc97a8e5 | ||
![]() |
3b837e1d54 | ||
![]() |
bb6c2050bc | ||
![]() |
082d4f9691 | ||
![]() |
153d68a9cd | ||
![]() |
0404faa856 | ||
![]() |
afbc2d6b8f | ||
![]() |
89ecc8bd2f | ||
![]() |
7f21a2b319 | ||
![]() |
e2f07f6723 | ||
![]() |
a475e143b7 | ||
![]() |
e50fd80b2e | ||
![]() |
68ea1abc05 | ||
![]() |
2e76b306c4 | ||
![]() |
ca3cac4ed3 | ||
![]() |
41852460e1 | ||
![]() |
9ec4e083d9 | ||
![]() |
9560a1c4a7 | ||
![]() |
4f5a47ace7 | ||
![]() |
01c4d662f2 | ||
![]() |
9bdda77e89 | ||
![]() |
194024edb9 | ||
![]() |
bef0d3a6a1 | ||
![]() |
47a024b795 | ||
![]() |
39847f9c9d | ||
![]() |
fa7bd28c92 | ||
![]() |
279f78e4a8 | ||
![]() |
8ec3cbdb33 | ||
![]() |
7449f7e73f | ||
![]() |
d680fde759 | ||
![]() |
f24f21ca91 | ||
![]() |
c8ea37eec0 | ||
![]() |
b71f452795 | ||
![]() |
7ea1ece169 | ||
![]() |
aece3a37c0 | ||
![]() |
705871f8dc | ||
![]() |
4a11975349 | ||
![]() |
4d3d27f2c4 | ||
![]() |
d784a30d42 | ||
![]() |
35f776284b | ||
![]() |
f659a6fe37 | ||
![]() |
ad53c99fc4 | ||
![]() |
fa0172d00c | ||
![]() |
ba77a88714 | ||
![]() |
9b39087102 | ||
![]() |
845411b48c | ||
![]() |
d715867b09 | ||
![]() |
0ca2cdfbed | ||
![]() |
a00961b9ef | ||
![]() |
701c188bab | ||
![]() |
e81002807f | ||
![]() |
0d1c72386e | ||
![]() |
c91779dffe | ||
![]() |
3853cc9214 | ||
![]() |
a66b3f6b80 | ||
![]() |
c97ec32343 | ||
![]() |
2abba7e445 | ||
![]() |
f887c27ad1 | ||
![]() |
6ee8d74899 | ||
![]() |
f196c72563 | ||
![]() |
419e564441 | ||
![]() |
de97b54c95 | ||
![]() |
07001f7b5c | ||
![]() |
bee17fce64 | ||
![]() |
718904a853 | ||
![]() |
e14d652651 | ||
![]() |
98ae5270ef | ||
![]() |
19ccf0ab40 | ||
![]() |
72af4a69d6 | ||
![]() |
fe50f4229c | ||
![]() |
6021bec5ee | ||
![]() |
7fcadc85fa | ||
![]() |
ca4de877c1 | ||
![]() |
1dfecf9618 | ||
![]() |
0a3505ed89 | ||
![]() |
33cbf7eabe | ||
![]() |
935d97ce1a | ||
![]() |
9f73f0ca8d | ||
![]() |
5d7f971a82 | ||
![]() |
d8cdbac15e | ||
![]() |
03b8c1348c | ||
![]() |
25a0be7672 | ||
![]() |
08f1ce2d54 | ||
![]() |
bea20d0495 |
@@ -4,8 +4,7 @@
|
|||||||
"plugin:@typescript-eslint/recommended",
|
"plugin:@typescript-eslint/recommended",
|
||||||
"plugin:wc/recommended",
|
"plugin:wc/recommended",
|
||||||
"plugin:lit/recommended",
|
"plugin:lit/recommended",
|
||||||
"prettier",
|
"prettier"
|
||||||
"prettier/@typescript-eslint"
|
|
||||||
],
|
],
|
||||||
"parser": "@typescript-eslint/parser",
|
"parser": "@typescript-eslint/parser",
|
||||||
"parserOptions": {
|
"parserOptions": {
|
||||||
@@ -29,9 +28,7 @@
|
|||||||
"__BUILD__": false,
|
"__BUILD__": false,
|
||||||
"__VERSION__": false,
|
"__VERSION__": false,
|
||||||
"__STATIC_PATH__": false,
|
"__STATIC_PATH__": false,
|
||||||
"Polymer": true,
|
"Polymer": true
|
||||||
"webkitSpeechRecognition": false,
|
|
||||||
"ResizeObserver": false
|
|
||||||
},
|
},
|
||||||
"env": {
|
"env": {
|
||||||
"browser": true,
|
"browser": true,
|
||||||
@@ -84,8 +81,29 @@
|
|||||||
"@typescript-eslint/no-unused-vars": 0,
|
"@typescript-eslint/no-unused-vars": 0,
|
||||||
"@typescript-eslint/explicit-function-return-type": 0,
|
"@typescript-eslint/explicit-function-return-type": 0,
|
||||||
"@typescript-eslint/explicit-module-boundary-types": 0,
|
"@typescript-eslint/explicit-module-boundary-types": 0,
|
||||||
"@typescript-eslint/no-shadow": ["error"]
|
"@typescript-eslint/no-shadow": ["error"],
|
||||||
|
"@typescript-eslint/naming-convention": [
|
||||||
|
0,
|
||||||
|
{
|
||||||
|
"selector": "default",
|
||||||
|
"format": ["camelCase", "snake_case"],
|
||||||
|
"leadingUnderscore": "allow",
|
||||||
|
"trailingUnderscore": "allow"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"selector": ["variable"],
|
||||||
|
"format": ["camelCase", "snake_case", "UPPER_CASE"],
|
||||||
|
"leadingUnderscore": "allow",
|
||||||
|
"trailingUnderscore": "allow"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"selector": "typeLike",
|
||||||
|
"format": ["PascalCase"]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"lit/attribute-value-entities": 0
|
||||||
},
|
},
|
||||||
"plugins": ["disable", "import", "lit", "prettier", "@typescript-eslint"],
|
"plugins": ["disable", "import", "lit", "prettier", "@typescript-eslint"],
|
||||||
"processor": "disable/disable"
|
"processor": "disable/disable",
|
||||||
|
"ignorePatterns": ["src/resources/lit-virtualizer/*"]
|
||||||
}
|
}
|
||||||
|
35
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
35
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
@@ -1,8 +1,6 @@
|
|||||||
name: Report a bug with the UI, Frontend or Lovelace
|
name: Report a bug with the UI, Frontend or Lovelace
|
||||||
about: Report an issue related to the Home Assistant frontend.
|
description: Report an issue related to the Home Assistant frontend.
|
||||||
labels: bug
|
labels: bug
|
||||||
title: ""
|
|
||||||
issue_body: true
|
|
||||||
body:
|
body:
|
||||||
- type: markdown
|
- type: markdown
|
||||||
attributes:
|
attributes:
|
||||||
@@ -97,11 +95,7 @@ body:
|
|||||||
If your issue is about how an entity is shown in the UI, please add the
|
If your issue is about how an entity is shown in the UI, please add the
|
||||||
state and attributes for all situations. You can find this information
|
state and attributes for all situations. You can find this information
|
||||||
at Developer Tools -> States.
|
at Developer Tools -> States.
|
||||||
value: |
|
render: txt
|
||||||
```yaml
|
|
||||||
# Paste your state here.
|
|
||||||
|
|
||||||
```
|
|
||||||
- type: textarea
|
- type: textarea
|
||||||
attributes:
|
attributes:
|
||||||
label: Problem-relevant frontend configuration
|
label: Problem-relevant frontend configuration
|
||||||
@@ -110,29 +104,18 @@ body:
|
|||||||
configuration of the used cards. Fill this out even if it seems
|
configuration of the used cards. Fill this out even if it seems
|
||||||
unimportant to you. Please be sure to remove personal information like
|
unimportant to you. Please be sure to remove personal information like
|
||||||
passwords, private URLs and other credentials.
|
passwords, private URLs and other credentials.
|
||||||
value: |
|
render: yaml
|
||||||
```yaml
|
|
||||||
# Paste your YAML here.
|
|
||||||
|
|
||||||
```
|
|
||||||
- type: textarea
|
- type: textarea
|
||||||
attributes:
|
attributes:
|
||||||
label: Javascript errors shown in your browser console/inspector
|
label: Javascript errors shown in your browser console/inspector
|
||||||
description: >
|
description: >
|
||||||
If you come across any Javascript or other error logs, e.g., in your
|
If you come across any Javascript or other error logs, e.g., in your
|
||||||
browser console/inspector please provide them.
|
browser console/inspector please provide them.
|
||||||
value: |
|
render: txt
|
||||||
```txt
|
- type: textarea
|
||||||
# Paste your logs here.
|
|
||||||
|
|
||||||
```
|
|
||||||
- type: markdown
|
|
||||||
attributes:
|
attributes:
|
||||||
value: |
|
label: Additional information
|
||||||
## Additional information
|
description: >
|
||||||
- type: markdown
|
|
||||||
attributes:
|
|
||||||
value: |
|
|
||||||
If you have any additional information for us, use the field below.
|
If you have any additional information for us, use the field below.
|
||||||
Please note, you can attach screenshots or screen recordings here,
|
Please note, you can attach screenshots or screen recordings here, by
|
||||||
by dragging and dropping files in the field below.
|
dragging and dropping files in the field below.
|
||||||
|
6
.github/workflows/ci.yaml
vendored
6
.github/workflows/ci.yaml
vendored
@@ -37,9 +37,11 @@ jobs:
|
|||||||
- name: Build resources
|
- name: Build resources
|
||||||
run: ./node_modules/.bin/gulp gen-icons-json build-translations gather-gallery-demos
|
run: ./node_modules/.bin/gulp gen-icons-json build-translations gather-gallery-demos
|
||||||
- name: Run eslint
|
- name: Run eslint
|
||||||
run: ./node_modules/.bin/eslint '{**/src,src}/**/*.{js,ts,html}' --ignore-path .gitignore
|
run: yarn run lint:eslint
|
||||||
- name: Run tsc
|
- name: Run tsc
|
||||||
run: ./node_modules/.bin/tsc
|
run: yarn run lint:types
|
||||||
|
- name: Run prettier
|
||||||
|
run: yarn run lint:prettier
|
||||||
test:
|
test:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
|
8
.github/workflows/release.yaml
vendored
8
.github/workflows/release.yaml
vendored
@@ -6,8 +6,7 @@ on:
|
|||||||
- published
|
- published
|
||||||
|
|
||||||
env:
|
env:
|
||||||
WHEELS_TAG: 3.7-alpine3.11
|
PYTHON_VERSION: 3.8
|
||||||
PYTHON_VERSION: 3.7
|
|
||||||
NODE_VERSION: 12.1
|
NODE_VERSION: 12.1
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
@@ -64,6 +63,9 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
arch: ["aarch64", "armhf", "armv7", "amd64", "i386"]
|
arch: ["aarch64", "armhf", "armv7", "amd64", "i386"]
|
||||||
|
tag:
|
||||||
|
- "3.8-alpine3.12"
|
||||||
|
- "3.9-alpine3.13"
|
||||||
steps:
|
steps:
|
||||||
- name: Download requirements.txt
|
- name: Download requirements.txt
|
||||||
uses: actions/download-artifact@v2
|
uses: actions/download-artifact@v2
|
||||||
@@ -73,7 +75,7 @@ jobs:
|
|||||||
- name: Build wheels
|
- name: Build wheels
|
||||||
uses: home-assistant/wheels@master
|
uses: home-assistant/wheels@master
|
||||||
with:
|
with:
|
||||||
tag: ${{ env.WHEELS_TAG }}
|
tag: ${{ matrix.tag }}
|
||||||
arch: ${{ matrix.arch }}
|
arch: ${{ matrix.arch }}
|
||||||
wheels-host: ${{ secrets.WHEELS_HOST }}
|
wheels-host: ${{ secrets.WHEELS_HOST }}
|
||||||
wheels-key: ${{ secrets.WHEELS_KEY }}
|
wheels-key: ${{ secrets.WHEELS_KEY }}
|
||||||
|
2
.github/workflows/translations.yaml
vendored
2
.github/workflows/translations.yaml
vendored
@@ -7,7 +7,7 @@ on:
|
|||||||
branches:
|
branches:
|
||||||
- dev
|
- dev
|
||||||
paths:
|
paths:
|
||||||
- translations/en.json
|
- src/translations/en.json
|
||||||
|
|
||||||
env:
|
env:
|
||||||
NODE_VERSION: 12
|
NODE_VERSION: 12
|
||||||
|
21
.gitignore
vendored
21
.gitignore
vendored
@@ -1,10 +1,17 @@
|
|||||||
|
.DS_Store
|
||||||
|
.reify-cache
|
||||||
|
|
||||||
|
# build
|
||||||
build
|
build
|
||||||
build-translations/*
|
build-translations/*
|
||||||
|
hass_frontend/*
|
||||||
|
dist
|
||||||
|
|
||||||
|
# yarn
|
||||||
|
.yarn
|
||||||
|
yarn-error.log
|
||||||
node_modules/*
|
node_modules/*
|
||||||
npm-debug.log
|
npm-debug.log
|
||||||
.DS_Store
|
|
||||||
hass_frontend/*
|
|
||||||
.reify-cache
|
|
||||||
|
|
||||||
# Python stuff
|
# Python stuff
|
||||||
*.py[cod]
|
*.py[cod]
|
||||||
@@ -14,11 +21,8 @@ hass_frontend/*
|
|||||||
# venv stuff
|
# venv stuff
|
||||||
pyvenv.cfg
|
pyvenv.cfg
|
||||||
pip-selfcheck.json
|
pip-selfcheck.json
|
||||||
venv
|
venv/*
|
||||||
.venv
|
.venv
|
||||||
lib
|
|
||||||
bin
|
|
||||||
dist
|
|
||||||
|
|
||||||
# vscode
|
# vscode
|
||||||
.vscode/*
|
.vscode/*
|
||||||
@@ -31,9 +35,8 @@ src/cast/dev_const.ts
|
|||||||
|
|
||||||
# Secrets
|
# Secrets
|
||||||
.lokalise_token
|
.lokalise_token
|
||||||
yarn-error.log
|
|
||||||
|
|
||||||
#asdf
|
# asdf
|
||||||
.tool-versions
|
.tool-versions
|
||||||
|
|
||||||
# Home Assistant config
|
# Home Assistant config
|
||||||
|
4
.mocharc.cjs
Normal file
4
.mocharc.cjs
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
module.exports = {
|
||||||
|
require: "test-mocha/testconf.js",
|
||||||
|
timeout: 10000,
|
||||||
|
};
|
@@ -1,3 +1,4 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-var-requires */
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
const env = require("./env.js");
|
const env = require("./env.js");
|
||||||
const paths = require("./paths.js");
|
const paths = require("./paths.js");
|
||||||
@@ -51,15 +52,16 @@ module.exports.terserOptions = (latestBuild) => ({
|
|||||||
|
|
||||||
module.exports.babelOptions = ({ latestBuild }) => ({
|
module.exports.babelOptions = ({ latestBuild }) => ({
|
||||||
babelrc: false,
|
babelrc: false,
|
||||||
|
compact: false,
|
||||||
presets: [
|
presets: [
|
||||||
!latestBuild && [
|
!latestBuild && [
|
||||||
require("@babel/preset-env").default,
|
"@babel/preset-env",
|
||||||
{
|
{
|
||||||
useBuiltIns: "entry",
|
useBuiltIns: "entry",
|
||||||
corejs: "3.6",
|
corejs: "3.6",
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
require("@babel/preset-typescript").default,
|
"@babel/preset-typescript",
|
||||||
].filter(Boolean),
|
].filter(Boolean),
|
||||||
plugins: [
|
plugins: [
|
||||||
// Part of ES2018. Converts {...a, b: 2} to Object.assign({}, a, {b: 2})
|
// Part of ES2018. Converts {...a, b: 2} to Object.assign({}, a, {b: 2})
|
||||||
@@ -72,23 +74,12 @@ module.exports.babelOptions = ({ latestBuild }) => ({
|
|||||||
"@babel/plugin-syntax-dynamic-import",
|
"@babel/plugin-syntax-dynamic-import",
|
||||||
"@babel/plugin-proposal-optional-chaining",
|
"@babel/plugin-proposal-optional-chaining",
|
||||||
"@babel/plugin-proposal-nullish-coalescing-operator",
|
"@babel/plugin-proposal-nullish-coalescing-operator",
|
||||||
[
|
["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }],
|
||||||
require("@babel/plugin-proposal-decorators").default,
|
["@babel/plugin-proposal-private-methods", { loose: true }],
|
||||||
{ decoratorsBeforeExport: true },
|
["@babel/plugin-proposal-class-properties", { loose: true }],
|
||||||
],
|
|
||||||
[
|
|
||||||
require("@babel/plugin-proposal-class-properties").default,
|
|
||||||
{ loose: true },
|
|
||||||
],
|
|
||||||
].filter(Boolean),
|
].filter(Boolean),
|
||||||
});
|
});
|
||||||
|
|
||||||
// Are already ES5, cause warnings when babelified.
|
|
||||||
module.exports.babelExclude = () => [
|
|
||||||
require.resolve("@mdi/js/mdi.js"),
|
|
||||||
require.resolve("hls.js"),
|
|
||||||
];
|
|
||||||
|
|
||||||
const outputPath = (outputRoot, latestBuild) =>
|
const outputPath = (outputRoot, latestBuild) =>
|
||||||
path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5");
|
path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5");
|
||||||
|
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-var-requires */
|
||||||
const fs = require("fs");
|
const fs = require("fs");
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
const paths = require("./paths.js");
|
const paths = require("./paths.js");
|
||||||
|
@@ -85,6 +85,11 @@ gulp.task("copy-translations-app", async () => {
|
|||||||
copyTranslations(staticDir);
|
copyTranslations(staticDir);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
gulp.task("copy-translations-supervisor", async () => {
|
||||||
|
const staticDir = paths.hassio_output_static;
|
||||||
|
copyTranslations(staticDir);
|
||||||
|
});
|
||||||
|
|
||||||
gulp.task("copy-static-app", async () => {
|
gulp.task("copy-static-app", async () => {
|
||||||
const staticDir = paths.app_output_static;
|
const staticDir = paths.app_output_static;
|
||||||
// Basic static files
|
// Basic static files
|
||||||
|
@@ -10,6 +10,8 @@ require("./gen-icons-json.js");
|
|||||||
require("./webpack.js");
|
require("./webpack.js");
|
||||||
require("./compress.js");
|
require("./compress.js");
|
||||||
require("./rollup.js");
|
require("./rollup.js");
|
||||||
|
require("./gather-static.js");
|
||||||
|
require("./translations.js");
|
||||||
|
|
||||||
gulp.task(
|
gulp.task(
|
||||||
"develop-hassio",
|
"develop-hassio",
|
||||||
@@ -20,6 +22,8 @@ gulp.task(
|
|||||||
"clean-hassio",
|
"clean-hassio",
|
||||||
"gen-icons-json",
|
"gen-icons-json",
|
||||||
"gen-index-hassio-dev",
|
"gen-index-hassio-dev",
|
||||||
|
"build-supervisor-translations",
|
||||||
|
"copy-translations-supervisor",
|
||||||
env.useRollup() ? "rollup-watch-hassio" : "webpack-watch-hassio"
|
env.useRollup() ? "rollup-watch-hassio" : "webpack-watch-hassio"
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
@@ -32,6 +36,8 @@ gulp.task(
|
|||||||
},
|
},
|
||||||
"clean-hassio",
|
"clean-hassio",
|
||||||
"gen-icons-json",
|
"gen-icons-json",
|
||||||
|
"build-supervisor-translations",
|
||||||
|
"copy-translations-supervisor",
|
||||||
env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio",
|
env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio",
|
||||||
"gen-index-hassio-prod",
|
"gen-index-hassio-prod",
|
||||||
...// Don't compress running tests
|
...// Don't compress running tests
|
||||||
|
@@ -266,6 +266,7 @@ gulp.task(taskName, function () {
|
|||||||
TRANSLATION_FRAGMENTS.forEach((fragment) => {
|
TRANSLATION_FRAGMENTS.forEach((fragment) => {
|
||||||
delete data.ui.panel[fragment];
|
delete data.ui.panel[fragment];
|
||||||
});
|
});
|
||||||
|
delete data.supervisor;
|
||||||
return data;
|
return data;
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
@@ -342,6 +343,62 @@ gulp.task(
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
gulp.task("build-translation-fragment-supervisor", function () {
|
||||||
|
return gulp
|
||||||
|
.src(fullDir + "/*.json")
|
||||||
|
.pipe(transform((data) => data.supervisor))
|
||||||
|
.pipe(gulp.dest(workDir + "/supervisor"));
|
||||||
|
});
|
||||||
|
|
||||||
|
gulp.task("build-translation-flatten-supervisor", function () {
|
||||||
|
return gulp
|
||||||
|
.src(workDir + "/supervisor/*.json")
|
||||||
|
.pipe(
|
||||||
|
transform(function (data) {
|
||||||
|
// Polymer.AppLocalizeBehavior requires flattened json
|
||||||
|
return flatten(data);
|
||||||
|
})
|
||||||
|
)
|
||||||
|
.pipe(gulp.dest(outDir));
|
||||||
|
});
|
||||||
|
|
||||||
|
gulp.task("build-translation-write-metadata", function writeMetadata() {
|
||||||
|
return gulp
|
||||||
|
.src(
|
||||||
|
[
|
||||||
|
path.join(paths.translations_src, "translationMetadata.json"),
|
||||||
|
workDir + "/testMetadata.json",
|
||||||
|
workDir + "/translationFingerprints.json",
|
||||||
|
],
|
||||||
|
{ allowEmpty: true }
|
||||||
|
)
|
||||||
|
.pipe(merge({}))
|
||||||
|
.pipe(
|
||||||
|
transform(function (data) {
|
||||||
|
const newData = {};
|
||||||
|
Object.entries(data).forEach(([key, value]) => {
|
||||||
|
// Filter out translations without native name.
|
||||||
|
if (value.nativeName) {
|
||||||
|
newData[key] = value;
|
||||||
|
} else {
|
||||||
|
console.warn(
|
||||||
|
`Skipping language ${key}. Native name was not translated.`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return newData;
|
||||||
|
})
|
||||||
|
)
|
||||||
|
.pipe(
|
||||||
|
transform((data) => ({
|
||||||
|
fragments: TRANSLATION_FRAGMENTS,
|
||||||
|
translations: data,
|
||||||
|
}))
|
||||||
|
)
|
||||||
|
.pipe(rename("translationMetadata.json"))
|
||||||
|
.pipe(gulp.dest(workDir));
|
||||||
|
});
|
||||||
|
|
||||||
gulp.task(
|
gulp.task(
|
||||||
"build-translations",
|
"build-translations",
|
||||||
gulp.series(
|
gulp.series(
|
||||||
@@ -353,41 +410,20 @@ gulp.task(
|
|||||||
gulp.parallel(...splitTasks),
|
gulp.parallel(...splitTasks),
|
||||||
"build-flattened-translations",
|
"build-flattened-translations",
|
||||||
"build-translation-fingerprints",
|
"build-translation-fingerprints",
|
||||||
function writeMetadata() {
|
"build-translation-write-metadata"
|
||||||
return gulp
|
)
|
||||||
.src(
|
);
|
||||||
[
|
|
||||||
path.join(paths.translations_src, "translationMetadata.json"),
|
gulp.task(
|
||||||
workDir + "/testMetadata.json",
|
"build-supervisor-translations",
|
||||||
workDir + "/translationFingerprints.json",
|
gulp.series(
|
||||||
],
|
"clean-translations",
|
||||||
{ allowEmpty: true }
|
"ensure-translations-build-dir",
|
||||||
)
|
"build-master-translation",
|
||||||
.pipe(merge({}))
|
"build-merged-translations",
|
||||||
.pipe(
|
"build-translation-fragment-supervisor",
|
||||||
transform(function (data) {
|
"build-translation-flatten-supervisor",
|
||||||
const newData = {};
|
"build-translation-fingerprints",
|
||||||
Object.entries(data).forEach(([key, value]) => {
|
"build-translation-write-metadata"
|
||||||
// Filter out translations without native name.
|
|
||||||
if (value.nativeName) {
|
|
||||||
newData[key] = value;
|
|
||||||
} else {
|
|
||||||
console.warn(
|
|
||||||
`Skipping language ${key}. Native name was not translated.`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return newData;
|
|
||||||
})
|
|
||||||
)
|
|
||||||
.pipe(
|
|
||||||
transform((data) => ({
|
|
||||||
fragments: TRANSLATION_FRAGMENTS,
|
|
||||||
translations: data,
|
|
||||||
}))
|
|
||||||
)
|
|
||||||
.pipe(rename("translationMetadata.json"))
|
|
||||||
.pipe(gulp.dest(workDir));
|
|
||||||
}
|
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@@ -137,7 +137,12 @@ gulp.task("webpack-watch-hassio", () => {
|
|||||||
isProdBuild: false,
|
isProdBuild: false,
|
||||||
latestBuild: true,
|
latestBuild: true,
|
||||||
})
|
})
|
||||||
).watch({}, doneHandler());
|
).watch({ ignored: /build-translations/ }, doneHandler());
|
||||||
|
|
||||||
|
gulp.watch(
|
||||||
|
path.join(paths.translations_src, "en.json"),
|
||||||
|
gulp.series("build-supervisor-translations", "copy-translations-supervisor")
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("webpack-prod-hassio", () =>
|
gulp.task("webpack-prod-hassio", () =>
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-var-requires */
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
@@ -34,6 +35,7 @@ module.exports = {
|
|||||||
|
|
||||||
hassio_dir: path.resolve(__dirname, "../hassio"),
|
hassio_dir: path.resolve(__dirname, "../hassio"),
|
||||||
hassio_output_root: path.resolve(__dirname, "../hassio/build"),
|
hassio_output_root: path.resolve(__dirname, "../hassio/build"),
|
||||||
|
hassio_output_static: path.resolve(__dirname, "../hassio/build/static"),
|
||||||
hassio_output_latest: path.resolve(
|
hassio_output_latest: path.resolve(
|
||||||
__dirname,
|
__dirname,
|
||||||
"../hassio/build/frontend_latest"
|
"../hassio/build/frontend_latest"
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-var-requires */
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
|
|
||||||
const commonjs = require("@rollup/plugin-commonjs");
|
const commonjs = require("@rollup/plugin-commonjs");
|
||||||
@@ -32,88 +33,77 @@ const createRollupConfig = ({
|
|||||||
publicPath,
|
publicPath,
|
||||||
dontHash,
|
dontHash,
|
||||||
isWDS,
|
isWDS,
|
||||||
}) => {
|
}) => ({
|
||||||
return {
|
/**
|
||||||
/**
|
* @type { import("rollup").InputOptions }
|
||||||
* @type { import("rollup").InputOptions }
|
*/
|
||||||
*/
|
inputOptions: {
|
||||||
inputOptions: {
|
input: entry,
|
||||||
input: entry,
|
// Some entry points contain no JavaScript. This setting silences a warning about that.
|
||||||
// Some entry points contain no JavaScript. This setting silences a warning about that.
|
// https://rollupjs.org/guide/en/#preserveentrysignatures
|
||||||
// https://rollupjs.org/guide/en/#preserveentrysignatures
|
preserveEntrySignatures: false,
|
||||||
preserveEntrySignatures: false,
|
plugins: [
|
||||||
plugins: [
|
ignore({
|
||||||
ignore({
|
files: bundle.emptyPackages({ latestBuild }),
|
||||||
files: bundle.emptyPackages({ latestBuild }),
|
}),
|
||||||
|
resolve({
|
||||||
|
extensions,
|
||||||
|
preferBuiltins: false,
|
||||||
|
browser: true,
|
||||||
|
rootDir: paths.polymer_dir,
|
||||||
|
}),
|
||||||
|
commonjs(),
|
||||||
|
json(),
|
||||||
|
babel({
|
||||||
|
...bundle.babelOptions({ latestBuild }),
|
||||||
|
extensions,
|
||||||
|
babelHelpers: isWDS ? "inline" : "bundled",
|
||||||
|
}),
|
||||||
|
string({
|
||||||
|
// Import certain extensions as strings
|
||||||
|
include: [path.join(paths.polymer_dir, "node_modules/**/*.css")],
|
||||||
|
}),
|
||||||
|
replace(bundle.definedVars({ isProdBuild, latestBuild, defineOverlay })),
|
||||||
|
!isWDS &&
|
||||||
|
manifest({
|
||||||
|
publicPath,
|
||||||
}),
|
}),
|
||||||
resolve({
|
!isWDS && worker(),
|
||||||
extensions,
|
!isWDS && dontHashPlugin({ dontHash }),
|
||||||
preferBuiltins: false,
|
!isWDS && isProdBuild && terser(bundle.terserOptions(latestBuild)),
|
||||||
browser: true,
|
!isWDS &&
|
||||||
rootDir: paths.polymer_dir,
|
isStatsBuild &&
|
||||||
|
visualizer({
|
||||||
|
// https://github.com/btd/rollup-plugin-visualizer#options
|
||||||
|
open: true,
|
||||||
|
sourcemap: true,
|
||||||
}),
|
}),
|
||||||
commonjs({
|
].filter(Boolean),
|
||||||
namedExports: {
|
},
|
||||||
"js-yaml": ["safeDump", "safeLoad"],
|
/**
|
||||||
},
|
* @type { import("rollup").OutputOptions }
|
||||||
}),
|
*/
|
||||||
json(),
|
outputOptions: {
|
||||||
babel({
|
// https://rollupjs.org/guide/en/#outputdir
|
||||||
...bundle.babelOptions({ latestBuild }),
|
dir: outputPath,
|
||||||
extensions,
|
// https://rollupjs.org/guide/en/#outputformat
|
||||||
exclude: bundle.babelExclude(),
|
format: latestBuild ? "es" : "systemjs",
|
||||||
babelHelpers: isWDS ? "inline" : "bundled",
|
// https://rollupjs.org/guide/en/#outputexternallivebindings
|
||||||
}),
|
externalLiveBindings: false,
|
||||||
string({
|
// https://rollupjs.org/guide/en/#outputentryfilenames
|
||||||
// Import certain extensions as strings
|
// https://rollupjs.org/guide/en/#outputchunkfilenames
|
||||||
include: [path.join(paths.polymer_dir, "node_modules/**/*.css")],
|
// https://rollupjs.org/guide/en/#outputassetfilenames
|
||||||
}),
|
entryFileNames:
|
||||||
replace(
|
isProdBuild && !isStatsBuild ? "[name]-[hash].js" : "[name].js",
|
||||||
bundle.definedVars({ isProdBuild, latestBuild, defineOverlay })
|
chunkFileNames: isProdBuild && !isStatsBuild ? "c.[hash].js" : "[name].js",
|
||||||
),
|
assetFileNames: isProdBuild && !isStatsBuild ? "a.[hash].js" : "[name].js",
|
||||||
!isWDS &&
|
// https://rollupjs.org/guide/en/#outputsourcemap
|
||||||
manifest({
|
sourcemap: isProdBuild ? true : "inline",
|
||||||
publicPath,
|
},
|
||||||
}),
|
});
|
||||||
!isWDS && worker(),
|
|
||||||
!isWDS && dontHashPlugin({ dontHash }),
|
|
||||||
!isWDS && isProdBuild && terser(bundle.terserOptions(latestBuild)),
|
|
||||||
!isWDS &&
|
|
||||||
isStatsBuild &&
|
|
||||||
visualizer({
|
|
||||||
// https://github.com/btd/rollup-plugin-visualizer#options
|
|
||||||
open: true,
|
|
||||||
sourcemap: true,
|
|
||||||
}),
|
|
||||||
].filter(Boolean),
|
|
||||||
},
|
|
||||||
/**
|
|
||||||
* @type { import("rollup").OutputOptions }
|
|
||||||
*/
|
|
||||||
outputOptions: {
|
|
||||||
// https://rollupjs.org/guide/en/#outputdir
|
|
||||||
dir: outputPath,
|
|
||||||
// https://rollupjs.org/guide/en/#outputformat
|
|
||||||
format: latestBuild ? "es" : "systemjs",
|
|
||||||
// https://rollupjs.org/guide/en/#outputexternallivebindings
|
|
||||||
externalLiveBindings: false,
|
|
||||||
// https://rollupjs.org/guide/en/#outputentryfilenames
|
|
||||||
// https://rollupjs.org/guide/en/#outputchunkfilenames
|
|
||||||
// https://rollupjs.org/guide/en/#outputassetfilenames
|
|
||||||
entryFileNames:
|
|
||||||
isProdBuild && !isStatsBuild ? "[name]-[hash].js" : "[name].js",
|
|
||||||
chunkFileNames:
|
|
||||||
isProdBuild && !isStatsBuild ? "c.[hash].js" : "[name].js",
|
|
||||||
assetFileNames:
|
|
||||||
isProdBuild && !isStatsBuild ? "a.[hash].js" : "[name].js",
|
|
||||||
// https://rollupjs.org/guide/en/#outputsourcemap
|
|
||||||
sourcemap: isProdBuild ? true : "inline",
|
|
||||||
},
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild, isWDS }) => {
|
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild, isWDS }) =>
|
||||||
return createRollupConfig(
|
createRollupConfig(
|
||||||
bundle.config.app({
|
bundle.config.app({
|
||||||
isProdBuild,
|
isProdBuild,
|
||||||
latestBuild,
|
latestBuild,
|
||||||
@@ -121,31 +111,24 @@ const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild, isWDS }) => {
|
|||||||
isWDS,
|
isWDS,
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
};
|
|
||||||
|
|
||||||
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => {
|
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
|
||||||
return createRollupConfig(
|
createRollupConfig(
|
||||||
bundle.config.demo({
|
bundle.config.demo({
|
||||||
isProdBuild,
|
isProdBuild,
|
||||||
latestBuild,
|
latestBuild,
|
||||||
isStatsBuild,
|
isStatsBuild,
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
};
|
|
||||||
|
|
||||||
const createCastConfig = ({ isProdBuild, latestBuild }) => {
|
const createCastConfig = ({ isProdBuild, latestBuild }) =>
|
||||||
return createRollupConfig(bundle.config.cast({ isProdBuild, latestBuild }));
|
createRollupConfig(bundle.config.cast({ isProdBuild, latestBuild }));
|
||||||
};
|
|
||||||
|
|
||||||
const createHassioConfig = ({ isProdBuild, latestBuild }) => {
|
const createHassioConfig = ({ isProdBuild, latestBuild }) =>
|
||||||
return createRollupConfig(bundle.config.hassio({ isProdBuild, latestBuild }));
|
createRollupConfig(bundle.config.hassio({ isProdBuild, latestBuild }));
|
||||||
};
|
|
||||||
|
|
||||||
const createGalleryConfig = ({ isProdBuild, latestBuild }) => {
|
const createGalleryConfig = ({ isProdBuild, latestBuild }) =>
|
||||||
return createRollupConfig(
|
createRollupConfig(bundle.config.gallery({ isProdBuild, latestBuild }));
|
||||||
bundle.config.gallery({ isProdBuild, latestBuild })
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
createAppConfig,
|
createAppConfig,
|
||||||
|
@@ -1,3 +1,4 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-var-requires */
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
const fs = require("fs");
|
const fs = require("fs");
|
||||||
|
|
||||||
|
@@ -1,9 +1,10 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-var-requires */
|
||||||
const webpack = require("webpack");
|
const webpack = require("webpack");
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
const TerserPlugin = require("terser-webpack-plugin");
|
const TerserPlugin = require("terser-webpack-plugin");
|
||||||
const { WebpackManifestPlugin } = require("webpack-manifest-plugin");
|
const { WebpackManifestPlugin } = require("webpack-manifest-plugin");
|
||||||
const paths = require("./paths.js");
|
const paths = require("./paths.js");
|
||||||
const bundle = require("./bundle");
|
const bundle = require("./bundle.js");
|
||||||
const log = require("fancy-log");
|
const log = require("fancy-log");
|
||||||
|
|
||||||
class LogStartCompilePlugin {
|
class LogStartCompilePlugin {
|
||||||
@@ -46,7 +47,6 @@ const createWebpackConfig = ({
|
|||||||
rules: [
|
rules: [
|
||||||
{
|
{
|
||||||
test: /\.m?js$|\.ts$/,
|
test: /\.m?js$|\.ts$/,
|
||||||
exclude: bundle.babelExclude(),
|
|
||||||
use: {
|
use: {
|
||||||
loader: "babel-loader",
|
loader: "babel-loader",
|
||||||
options: bundle.babelOptions({ latestBuild }),
|
options: bundle.babelOptions({ latestBuild }),
|
||||||
@@ -94,6 +94,7 @@ const createWebpackConfig = ({
|
|||||||
? path.resolve(context, resource)
|
? path.resolve(context, resource)
|
||||||
: require.resolve(resource);
|
: require.resolve(resource);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
console.error(
|
console.error(
|
||||||
"Error in Home Assistant ignore plugin",
|
"Error in Home Assistant ignore plugin",
|
||||||
resource,
|
resource,
|
||||||
@@ -114,8 +115,9 @@ const createWebpackConfig = ({
|
|||||||
// We need to change the import of the polyfill for EventTarget, so we replace the polyfill file with our customized one
|
// We need to change the import of the polyfill for EventTarget, so we replace the polyfill file with our customized one
|
||||||
new webpack.NormalModuleReplacementPlugin(
|
new webpack.NormalModuleReplacementPlugin(
|
||||||
new RegExp(
|
new RegExp(
|
||||||
require.resolve(
|
path.resolve(
|
||||||
"lit-virtualizer/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js"
|
paths.polymer_dir,
|
||||||
|
"src/resources/lit-virtualizer/lib/uni-virtualizer/lib/polyfillLoaders/EventTarget.js"
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
path.resolve(paths.polymer_dir, "src/resources/EventTarget-ponyfill.js")
|
path.resolve(paths.polymer_dir, "src/resources/EventTarget-ponyfill.js")
|
||||||
@@ -124,6 +126,11 @@ const createWebpackConfig = ({
|
|||||||
].filter(Boolean),
|
].filter(Boolean),
|
||||||
resolve: {
|
resolve: {
|
||||||
extensions: [".ts", ".js", ".json"],
|
extensions: [".ts", ".js", ".json"],
|
||||||
|
alias: {
|
||||||
|
"lit/decorators$": "lit/decorators.js",
|
||||||
|
"lit/directive$": "lit/directive.js",
|
||||||
|
"lit/polyfill-support$": "lit/polyfill-support.js",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
output: {
|
output: {
|
||||||
filename: ({ chunk }) => {
|
filename: ({ chunk }) => {
|
||||||
@@ -144,33 +151,24 @@ const createWebpackConfig = ({
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => {
|
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
|
||||||
return createWebpackConfig(
|
createWebpackConfig(
|
||||||
bundle.config.app({ isProdBuild, latestBuild, isStatsBuild })
|
bundle.config.app({ isProdBuild, latestBuild, isStatsBuild })
|
||||||
);
|
);
|
||||||
};
|
|
||||||
|
|
||||||
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => {
|
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
|
||||||
return createWebpackConfig(
|
createWebpackConfig(
|
||||||
bundle.config.demo({ isProdBuild, latestBuild, isStatsBuild })
|
bundle.config.demo({ isProdBuild, latestBuild, isStatsBuild })
|
||||||
);
|
);
|
||||||
};
|
|
||||||
|
|
||||||
const createCastConfig = ({ isProdBuild, latestBuild }) => {
|
const createCastConfig = ({ isProdBuild, latestBuild }) =>
|
||||||
return createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild }));
|
createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild }));
|
||||||
};
|
|
||||||
|
|
||||||
const createHassioConfig = ({ isProdBuild, latestBuild }) => {
|
const createHassioConfig = ({ isProdBuild, latestBuild }) =>
|
||||||
return createWebpackConfig(
|
createWebpackConfig(bundle.config.hassio({ isProdBuild, latestBuild }));
|
||||||
bundle.config.hassio({ isProdBuild, latestBuild })
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const createGalleryConfig = ({ isProdBuild, latestBuild }) => {
|
const createGalleryConfig = ({ isProdBuild, latestBuild }) =>
|
||||||
return createWebpackConfig(
|
createWebpackConfig(bundle.config.gallery({ isProdBuild, latestBuild }));
|
||||||
bundle.config.gallery({ isProdBuild, latestBuild })
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
createAppConfig,
|
createAppConfig,
|
||||||
|
@@ -1,16 +1,9 @@
|
|||||||
|
import "@material/mwc-button/mwc-button";
|
||||||
import "@polymer/paper-item/paper-icon-item";
|
import "@polymer/paper-item/paper-icon-item";
|
||||||
import "@polymer/paper-listbox/paper-listbox";
|
import "@polymer/paper-listbox/paper-listbox";
|
||||||
import { Auth, Connection } from "home-assistant-js-websocket";
|
import { Auth, Connection } from "home-assistant-js-websocket";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property, state } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
internalProperty,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { CastManager } from "../../../../src/cast/cast_manager";
|
import { CastManager } from "../../../../src/cast/cast_manager";
|
||||||
import {
|
import {
|
||||||
castSendShowLovelaceView,
|
castSendShowLovelaceView,
|
||||||
@@ -32,7 +25,6 @@ import {
|
|||||||
import "../../../../src/layouts/hass-loading-screen";
|
import "../../../../src/layouts/hass-loading-screen";
|
||||||
import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config";
|
import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config";
|
||||||
import "./hc-layout";
|
import "./hc-layout";
|
||||||
import "@material/mwc-button/mwc-button";
|
|
||||||
|
|
||||||
@customElement("hc-cast")
|
@customElement("hc-cast")
|
||||||
class HcCast extends LitElement {
|
class HcCast extends LitElement {
|
||||||
@@ -42,9 +34,9 @@ class HcCast extends LitElement {
|
|||||||
|
|
||||||
@property() public castManager!: CastManager;
|
@property() public castManager!: CastManager;
|
||||||
|
|
||||||
@internalProperty() private askWrite = false;
|
@state() private askWrite = false;
|
||||||
|
|
||||||
@internalProperty() private lovelaceConfig?: LovelaceConfig | null;
|
@state() private lovelaceConfig?: LovelaceConfig | null;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
if (this.lovelaceConfig === undefined) {
|
if (this.lovelaceConfig === undefined) {
|
||||||
@@ -54,9 +46,7 @@ class HcCast extends LitElement {
|
|||||||
const error =
|
const error =
|
||||||
this.castManager.castState === "NO_DEVICES_AVAILABLE"
|
this.castManager.castState === "NO_DEVICES_AVAILABLE"
|
||||||
? html`
|
? html`
|
||||||
<p>
|
<p>There were no suitable Chromecast devices to cast to found.</p>
|
||||||
There were no suitable Chromecast devices to cast to found.
|
|
||||||
</p>
|
|
||||||
`
|
`
|
||||||
: undefined;
|
: undefined;
|
||||||
|
|
||||||
@@ -206,7 +196,7 @@ class HcCast extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
.center-item {
|
.center-item {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
@@ -11,15 +11,8 @@ import {
|
|||||||
getAuth,
|
getAuth,
|
||||||
getAuthOptions,
|
getAuthOptions,
|
||||||
} from "home-assistant-js-websocket";
|
} from "home-assistant-js-websocket";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, state } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
TemplateResult,
|
|
||||||
internalProperty,
|
|
||||||
} from "lit-element";
|
|
||||||
import { CastManager, getCastManager } from "../../../../src/cast/cast_manager";
|
import { CastManager, getCastManager } from "../../../../src/cast/cast_manager";
|
||||||
import { castSendShowDemo } from "../../../../src/cast/receiver_messages";
|
import { castSendShowDemo } from "../../../../src/cast/receiver_messages";
|
||||||
import {
|
import {
|
||||||
@@ -60,19 +53,19 @@ const INTRO = html`
|
|||||||
|
|
||||||
@customElement("hc-connect")
|
@customElement("hc-connect")
|
||||||
export class HcConnect extends LitElement {
|
export class HcConnect extends LitElement {
|
||||||
@internalProperty() private loading = false;
|
@state() private loading = false;
|
||||||
|
|
||||||
// If we had stored credentials but we cannot connect,
|
// If we had stored credentials but we cannot connect,
|
||||||
// show a screen asking retry or logout.
|
// show a screen asking retry or logout.
|
||||||
@internalProperty() private cannotConnect = false;
|
@state() private cannotConnect = false;
|
||||||
|
|
||||||
@internalProperty() private error?: string | TemplateResult;
|
@state() private error?: string | TemplateResult;
|
||||||
|
|
||||||
@internalProperty() private auth?: Auth;
|
@state() private auth?: Auth;
|
||||||
|
|
||||||
@internalProperty() private connection?: Connection;
|
@state() private connection?: Connection;
|
||||||
|
|
||||||
@internalProperty() private castManager?: CastManager | null;
|
@state() private castManager?: CastManager | null;
|
||||||
|
|
||||||
private openDemo = false;
|
private openDemo = false;
|
||||||
|
|
||||||
@@ -86,9 +79,7 @@ export class HcConnect extends LitElement {
|
|||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
<a href="/">
|
<a href="/">
|
||||||
<mwc-button>
|
<mwc-button> Retry </mwc-button>
|
||||||
Retry
|
|
||||||
</mwc-button>
|
|
||||||
</a>
|
</a>
|
||||||
<div class="spacer"></div>
|
<div class="spacer"></div>
|
||||||
<mwc-button @click=${this._handleLogout}>Log out</mwc-button>
|
<mwc-button @click=${this._handleLogout}>Log out</mwc-button>
|
||||||
@@ -299,7 +290,7 @@ export class HcConnect extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
.card-content a {
|
.card-content a {
|
||||||
color: var(--primary-color);
|
color: var(--primary-color);
|
||||||
|
@@ -4,15 +4,8 @@ import {
|
|||||||
getUser,
|
getUser,
|
||||||
HassUser,
|
HassUser,
|
||||||
} from "home-assistant-js-websocket";
|
} from "home-assistant-js-websocket";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
|
|
||||||
@customElement("hc-layout")
|
@customElement("hc-layout")
|
||||||
@@ -69,7 +62,7 @@ class HcLayout extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
:host {
|
:host {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
@@ -1,10 +1,5 @@
|
|||||||
import {
|
import { html, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, property, state } from "lit/decorators";
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { mockHistory } from "../../../../demo/src/stubs/history";
|
import { mockHistory } from "../../../../demo/src/stubs/history";
|
||||||
import { LovelaceConfig } from "../../../../src/data/lovelace";
|
import { LovelaceConfig } from "../../../../src/data/lovelace";
|
||||||
import {
|
import {
|
||||||
@@ -21,7 +16,7 @@ import "./hc-lovelace";
|
|||||||
class HcDemo extends HassElement {
|
class HcDemo extends HassElement {
|
||||||
@property({ attribute: false }) public lovelacePath!: string;
|
@property({ attribute: false }) public lovelacePath!: string;
|
||||||
|
|
||||||
@internalProperty() private _lovelaceConfig?: LovelaceConfig;
|
@state() private _lovelaceConfig?: LovelaceConfig;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
if (!this._lovelaceConfig) {
|
if (!this._lovelaceConfig) {
|
||||||
@@ -38,10 +33,10 @@ class HcDemo extends HassElement {
|
|||||||
|
|
||||||
protected firstUpdated(changedProps) {
|
protected firstUpdated(changedProps) {
|
||||||
super.firstUpdated(changedProps);
|
super.firstUpdated(changedProps);
|
||||||
this._initialize();
|
this._initializeHass();
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _initialize() {
|
private async _initializeHass() {
|
||||||
const initial: Partial<MockHomeAssistant> = {
|
const initial: Partial<MockHomeAssistant> = {
|
||||||
// Override updateHass so that the correct hass lifecycle methods are called
|
// Override updateHass so that the correct hass lifecycle methods are called
|
||||||
updateHass: (hassUpdate: Partial<HomeAssistant>) =>
|
updateHass: (hassUpdate: Partial<HomeAssistant>) =>
|
||||||
|
@@ -1,12 +1,5 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
|
|
||||||
@customElement("hc-launch-screen")
|
@customElement("hc-launch-screen")
|
||||||
@@ -29,7 +22,7 @@ class HcLaunchScreen extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
:host {
|
:host {
|
||||||
display: block;
|
display: block;
|
||||||
|
@@ -1,12 +1,5 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { LovelaceConfig } from "../../../../src/data/lovelace";
|
import { LovelaceConfig } from "../../../../src/data/lovelace";
|
||||||
import { Lovelace } from "../../../../src/panels/lovelace/types";
|
import { Lovelace } from "../../../../src/panels/lovelace/types";
|
||||||
import "../../../../src/panels/lovelace/views/hui-view";
|
import "../../../../src/panels/lovelace/views/hui-view";
|
||||||
@@ -35,11 +28,12 @@ class HcLovelace extends LitElement {
|
|||||||
}
|
}
|
||||||
const lovelace: Lovelace = {
|
const lovelace: Lovelace = {
|
||||||
config: this.lovelaceConfig,
|
config: this.lovelaceConfig,
|
||||||
|
rawConfig: this.lovelaceConfig,
|
||||||
editMode: false,
|
editMode: false,
|
||||||
urlPath: this.urlPath!,
|
urlPath: this.urlPath!,
|
||||||
enableFullEditMode: () => undefined,
|
enableFullEditMode: () => undefined,
|
||||||
mode: "storage",
|
mode: "storage",
|
||||||
language: "en",
|
locale: this.hass.locale,
|
||||||
saveConfig: async () => undefined,
|
saveConfig: async () => undefined,
|
||||||
deleteConfig: async () => undefined,
|
deleteConfig: async () => undefined,
|
||||||
setEditMode: () => undefined,
|
setEditMode: () => undefined,
|
||||||
@@ -90,10 +84,11 @@ class HcLovelace extends LitElement {
|
|||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
:host {
|
:host {
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
|
height: 0;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
|
@@ -3,12 +3,8 @@ import {
|
|||||||
getAuth,
|
getAuth,
|
||||||
UnsubscribeFunc,
|
UnsubscribeFunc,
|
||||||
} from "home-assistant-js-websocket";
|
} from "home-assistant-js-websocket";
|
||||||
import {
|
import { html, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, state } from "lit/decorators";
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { CAST_NS } from "../../../../src/cast/const";
|
import { CAST_NS } from "../../../../src/cast/const";
|
||||||
import {
|
import {
|
||||||
ConnectMessage,
|
ConnectMessage,
|
||||||
@@ -36,13 +32,13 @@ let resourcesLoaded = false;
|
|||||||
|
|
||||||
@customElement("hc-main")
|
@customElement("hc-main")
|
||||||
export class HcMain extends HassElement {
|
export class HcMain extends HassElement {
|
||||||
@internalProperty() private _showDemo = false;
|
@state() private _showDemo = false;
|
||||||
|
|
||||||
@internalProperty() private _lovelaceConfig?: LovelaceConfig;
|
@state() private _lovelaceConfig?: LovelaceConfig;
|
||||||
|
|
||||||
@internalProperty() private _lovelacePath: string | number | null = null;
|
@state() private _lovelacePath: string | number | null = null;
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
@state() private _error?: string;
|
||||||
|
|
||||||
private _unsubLovelace?: UnsubscribeFunc;
|
private _unsubLovelace?: UnsubscribeFunc;
|
||||||
|
|
||||||
@@ -221,11 +217,17 @@ export class HcMain extends HassElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async _generateLovelaceConfig() {
|
private async _generateLovelaceConfig() {
|
||||||
const { generateLovelaceConfigFromHass } = await import(
|
const { generateLovelaceDashboardStrategy } = await import(
|
||||||
"../../../../src/panels/lovelace/common/generate-lovelace-config"
|
"../../../../src/panels/lovelace/strategies/get-strategy"
|
||||||
);
|
);
|
||||||
this._handleNewLovelaceConfig(
|
this._handleNewLovelaceConfig(
|
||||||
await generateLovelaceConfigFromHass(this.hass!)
|
await generateLovelaceDashboardStrategy(
|
||||||
|
{
|
||||||
|
hass: this.hass!,
|
||||||
|
narrow: false,
|
||||||
|
},
|
||||||
|
"original-states"
|
||||||
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -246,11 +246,15 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
|
|||||||
|
|
||||||
"light.living_room_lights": {
|
"light.living_room_lights": {
|
||||||
entity_id: "light.living_room_lights",
|
entity_id: "light.living_room_lights",
|
||||||
state: "off",
|
state: "on",
|
||||||
attributes: {
|
attributes: {
|
||||||
min_mireds: 111,
|
min_mireds: 111,
|
||||||
max_mireds: 400,
|
max_mireds: 400,
|
||||||
|
brightness: 175,
|
||||||
|
color_temp: 300,
|
||||||
|
supported_color_modes: ["brightness", "color_temp"],
|
||||||
friendly_name: "Living Room Lights",
|
friendly_name: "Living Room Lights",
|
||||||
|
color_mode: "color_temp",
|
||||||
supported_features: 55,
|
supported_features: 55,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@@ -263,13 +267,27 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
|
|||||||
},
|
},
|
||||||
"light.kitchen_lights": {
|
"light.kitchen_lights": {
|
||||||
entity_id: "light.kitchen_lights",
|
entity_id: "light.kitchen_lights",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
min_mireds: 111,
|
||||||
|
max_mireds: 400,
|
||||||
|
brightness: 200,
|
||||||
|
rgb_color: [255, 175, 96],
|
||||||
|
supported_color_modes: ["brightness", "color_temp", "rgb"],
|
||||||
|
color_mode: "rgb",
|
||||||
|
friendly_name: "Kitchen Lights",
|
||||||
|
supported_features: 55,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"light.lifx5": {
|
||||||
|
entity_id: "light.lifx5",
|
||||||
state: "off",
|
state: "off",
|
||||||
attributes: {
|
attributes: {
|
||||||
friendly_name: "Kitchen Lights",
|
supported_color_modes: ["brightness"],
|
||||||
|
friendly_name: "Garage Lights",
|
||||||
supported_features: 1,
|
supported_features: 1,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
"sensor.plexspy": {
|
"sensor.plexspy": {
|
||||||
entity_id: "sensor.plexspy",
|
entity_id: "sensor.plexspy",
|
||||||
state: "0",
|
state: "0",
|
||||||
@@ -482,16 +500,6 @@ export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
|
|||||||
icon: "hademo:history",
|
icon: "hademo:history",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
"light.lifx5": {
|
|
||||||
entity_id: "light.lifx5",
|
|
||||||
state: "on",
|
|
||||||
attributes: {
|
|
||||||
min_mireds: 111,
|
|
||||||
max_mireds: 400,
|
|
||||||
friendly_name: "Garage Lights",
|
|
||||||
supported_features: 55,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.alok_to_home": {
|
"sensor.alok_to_home": {
|
||||||
entity_id: "sensor.alok_to_home",
|
entity_id: "sensor.alok_to_home",
|
||||||
state: "41",
|
state: "41",
|
||||||
|
@@ -1114,6 +1114,9 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
min_mireds: 153,
|
min_mireds: 153,
|
||||||
max_mireds: 500,
|
max_mireds: 500,
|
||||||
brightness: 63,
|
brightness: 63,
|
||||||
|
color_temp: 200,
|
||||||
|
supported_color_modes: ["brightness", "color_temp", "rgb"],
|
||||||
|
color_mode: "color_temp",
|
||||||
friendly_name: "Upstairs lights",
|
friendly_name: "Upstairs lights",
|
||||||
supported_features: 63,
|
supported_features: 63,
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
@@ -1125,6 +1128,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
attributes: {
|
attributes: {
|
||||||
friendly_name: "Walk in closet lights",
|
friendly_name: "Walk in closet lights",
|
||||||
supported_features: 41,
|
supported_features: 41,
|
||||||
|
supported_color_modes: ["brightness", "color_temp"],
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:wall-sconce",
|
icon: "mdi:wall-sconce",
|
||||||
},
|
},
|
||||||
@@ -1136,6 +1140,8 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
brightness: 254,
|
brightness: 254,
|
||||||
friendly_name: "Outdoor lights",
|
friendly_name: "Outdoor lights",
|
||||||
supported_features: 41,
|
supported_features: 41,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
|
color_mode: "brightness",
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:wall-sconce",
|
icon: "mdi:wall-sconce",
|
||||||
},
|
},
|
||||||
@@ -1148,6 +1154,8 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
max_mireds: 500,
|
max_mireds: 500,
|
||||||
brightness: 128,
|
brightness: 128,
|
||||||
color_temp: 366,
|
color_temp: 366,
|
||||||
|
supported_color_modes: ["brightness", "color_temp", "rgb"],
|
||||||
|
color_mode: "color_temp",
|
||||||
effect_list: ["colorloop"],
|
effect_list: ["colorloop"],
|
||||||
friendly_name: "Downstairs lights",
|
friendly_name: "Downstairs lights",
|
||||||
supported_features: 63,
|
supported_features: 63,
|
||||||
@@ -1307,6 +1315,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
attributes: {
|
attributes: {
|
||||||
min_mireds: 153,
|
min_mireds: 153,
|
||||||
max_mireds: 500,
|
max_mireds: 500,
|
||||||
|
supported_color_modes: ["brightness", "color_temp"],
|
||||||
is_deconz_group: false,
|
is_deconz_group: false,
|
||||||
friendly_name: "Bedside Lamp",
|
friendly_name: "Bedside Lamp",
|
||||||
supported_features: 63,
|
supported_features: 63,
|
||||||
@@ -1320,6 +1329,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
attributes: {
|
attributes: {
|
||||||
min_mireds: 153,
|
min_mireds: 153,
|
||||||
max_mireds: 500,
|
max_mireds: 500,
|
||||||
|
supported_color_modes: ["brightness", "color_temp"],
|
||||||
is_deconz_group: false,
|
is_deconz_group: false,
|
||||||
friendly_name: "Floorlamp Reading Light",
|
friendly_name: "Floorlamp Reading Light",
|
||||||
supported_features: 43,
|
supported_features: 43,
|
||||||
@@ -1335,6 +1345,8 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
max_mireds: 500,
|
max_mireds: 500,
|
||||||
brightness: 128,
|
brightness: 128,
|
||||||
color_temp: 366,
|
color_temp: 366,
|
||||||
|
supported_color_modes: ["brightness", "color_temp", "rgb"],
|
||||||
|
color_mode: "color_temp",
|
||||||
effect_list: ["colorloop"],
|
effect_list: ["colorloop"],
|
||||||
is_deconz_group: false,
|
is_deconz_group: false,
|
||||||
friendly_name: "Hallway window light",
|
friendly_name: "Hallway window light",
|
||||||
@@ -1349,6 +1361,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
attributes: {
|
attributes: {
|
||||||
brightness: 77,
|
brightness: 77,
|
||||||
is_deconz_group: false,
|
is_deconz_group: false,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
friendly_name: "Isa Ceiling Light",
|
friendly_name: "Isa Ceiling Light",
|
||||||
supported_features: 41,
|
supported_features: 41,
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
@@ -1363,6 +1376,8 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
max_mireds: 500,
|
max_mireds: 500,
|
||||||
brightness: 150,
|
brightness: 150,
|
||||||
color_temp: 366,
|
color_temp: 366,
|
||||||
|
supported_color_modes: ["brightness", "color_temp"],
|
||||||
|
color_mode: "color_temp",
|
||||||
effect_list: ["colorloop"],
|
effect_list: ["colorloop"],
|
||||||
is_deconz_group: false,
|
is_deconz_group: false,
|
||||||
friendly_name: "Floorlamp",
|
friendly_name: "Floorlamp",
|
||||||
@@ -1377,6 +1392,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
attributes: {
|
attributes: {
|
||||||
friendly_name: "Bedroom Ceiling Light",
|
friendly_name: "Bedroom Ceiling Light",
|
||||||
supported_features: 41,
|
supported_features: 41,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:ceiling-light",
|
icon: "mdi:ceiling-light",
|
||||||
},
|
},
|
||||||
@@ -1387,6 +1403,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
attributes: {
|
attributes: {
|
||||||
friendly_name: "Nightlight",
|
friendly_name: "Nightlight",
|
||||||
supported_features: 17,
|
supported_features: 17,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:lamp",
|
icon: "mdi:lamp",
|
||||||
},
|
},
|
||||||
@@ -1753,6 +1770,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
power_consumption: 2.2,
|
power_consumption: 2.2,
|
||||||
friendly_name: "Upstairs Hallway Light",
|
friendly_name: "Upstairs Hallway Light",
|
||||||
supported_features: 33,
|
supported_features: 33,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:ceiling-light",
|
icon: "mdi:ceiling-light",
|
||||||
},
|
},
|
||||||
@@ -1768,6 +1786,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
power_consumption: 0,
|
power_consumption: 0,
|
||||||
friendly_name: "Dining Room Light",
|
friendly_name: "Dining Room Light",
|
||||||
supported_features: 33,
|
supported_features: 33,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:ceiling-light",
|
icon: "mdi:ceiling-light",
|
||||||
},
|
},
|
||||||
@@ -1783,6 +1802,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
power_consumption: 0,
|
power_consumption: 0,
|
||||||
friendly_name: "Living room Spotlights",
|
friendly_name: "Living room Spotlights",
|
||||||
supported_features: 33,
|
supported_features: 33,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:track-light",
|
icon: "mdi:track-light",
|
||||||
},
|
},
|
||||||
@@ -1799,6 +1819,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
power_consumption: 2.5,
|
power_consumption: 2.5,
|
||||||
friendly_name: "Passage Lights",
|
friendly_name: "Passage Lights",
|
||||||
supported_features: 33,
|
supported_features: 33,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:track-light",
|
icon: "mdi:track-light",
|
||||||
},
|
},
|
||||||
@@ -1843,6 +1864,7 @@ export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
|||||||
power_consumption: 37.4,
|
power_consumption: 37.4,
|
||||||
friendly_name: "Kitchen Lights",
|
friendly_name: "Kitchen Lights",
|
||||||
supported_features: 33,
|
supported_features: 33,
|
||||||
|
supported_color_modes: ["brightness"],
|
||||||
custom_ui_state_card: "state-card-custom-ui",
|
custom_ui_state_card: "state-card-custom-ui",
|
||||||
icon: "mdi:track-light",
|
icon: "mdi:track-light",
|
||||||
},
|
},
|
||||||
|
@@ -440,57 +440,43 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({
|
|||||||
type: "horizontal-stack",
|
type: "horizontal-stack",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
type: "grid",
|
||||||
|
columns: 2,
|
||||||
cards: [
|
cards: [
|
||||||
{
|
{
|
||||||
cards: [
|
graph: "line",
|
||||||
{
|
type: "sensor",
|
||||||
graph: "line",
|
entity: "sensor.temperature_bedroom",
|
||||||
type: "sensor",
|
|
||||||
entity: "sensor.temperature_bedroom",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
graph: "line",
|
|
||||||
type: "sensor",
|
|
||||||
name: "S's room",
|
|
||||||
entity: "sensor.temperature_stefan",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
type: "horizontal-stack",
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
cards: [
|
graph: "line",
|
||||||
{
|
type: "sensor",
|
||||||
graph: "line",
|
name: "S's room",
|
||||||
type: "sensor",
|
entity: "sensor.temperature_stefan",
|
||||||
entity: "sensor.temperature_passage",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
graph: "line",
|
|
||||||
type: "sensor",
|
|
||||||
name: "Bathroom",
|
|
||||||
entity: "sensor.temperature_downstairs_bathroom",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
type: "horizontal-stack",
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
cards: [
|
graph: "line",
|
||||||
{
|
type: "sensor",
|
||||||
graph: "line",
|
entity: "sensor.temperature_passage",
|
||||||
type: "sensor",
|
},
|
||||||
entity: "sensor.temperature_storage",
|
{
|
||||||
},
|
graph: "line",
|
||||||
{
|
type: "sensor",
|
||||||
graph: "line",
|
name: "Bathroom",
|
||||||
type: "sensor",
|
entity: "sensor.temperature_downstairs_bathroom",
|
||||||
name: "Refrigerator",
|
},
|
||||||
entity: "sensor.refrigerator",
|
{
|
||||||
},
|
graph: "line",
|
||||||
],
|
type: "sensor",
|
||||||
type: "horizontal-stack",
|
entity: "sensor.temperature_storage",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
graph: "line",
|
||||||
|
type: "sensor",
|
||||||
|
name: "Refrigerator",
|
||||||
|
entity: "sensor.refrigerator",
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
type: "vertical-stack",
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
entities: [
|
entities: [
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
/* eslint-disable */
|
/* eslint-disable */
|
||||||
import { LitElement } from "lit-element";
|
import { LitElement } from "lit";
|
||||||
import "./card-tools";
|
import "./card-tools";
|
||||||
|
|
||||||
class CardModder extends LitElement {
|
class CardModder extends LitElement {
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
/* eslint-disable */
|
/* eslint-disable */
|
||||||
import { html, LitElement } from "lit-element";
|
import { html, LitElement } from "lit";
|
||||||
|
|
||||||
if (!window.cardTools) {
|
if (!window.cardTools) {
|
||||||
const version = 0.2;
|
const version = 0.2;
|
||||||
|
@@ -1,12 +1,5 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, state } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { CastManager } from "../../../src/cast/cast_manager";
|
import { CastManager } from "../../../src/cast/cast_manager";
|
||||||
import { castSendShowDemo } from "../../../src/cast/receiver_messages";
|
import { castSendShowDemo } from "../../../src/cast/receiver_messages";
|
||||||
import "../../../src/components/ha-icon";
|
import "../../../src/components/ha-icon";
|
||||||
@@ -20,7 +13,7 @@ import { HomeAssistant } from "../../../src/types";
|
|||||||
class CastDemoRow extends LitElement implements LovelaceRow {
|
class CastDemoRow extends LitElement implements LovelaceRow {
|
||||||
public hass!: HomeAssistant;
|
public hass!: HomeAssistant;
|
||||||
|
|
||||||
@internalProperty() private _castManager?: CastManager | null;
|
@state() private _castManager?: CastManager | null;
|
||||||
|
|
||||||
public setConfig(_config: CastConfig): void {
|
public setConfig(_config: CastConfig): void {
|
||||||
// No config possible.
|
// No config possible.
|
||||||
@@ -73,7 +66,7 @@ class CastDemoRow extends LitElement implements LovelaceRow {
|
|||||||
this.style.display = this._castManager ? "" : "none";
|
this.style.display = this._castManager ? "" : "none";
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
:host {
|
:host {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
@@ -1,14 +1,7 @@
|
|||||||
import "@material/mwc-button";
|
import "@material/mwc-button";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { property, state } from "lit/decorators";
|
||||||
CSSResult,
|
import { until } from "lit/directives/until";
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { until } from "lit-html/directives/until";
|
|
||||||
import "../../../src/components/ha-card";
|
import "../../../src/components/ha-card";
|
||||||
import "../../../src/components/ha-circular-progress";
|
import "../../../src/components/ha-circular-progress";
|
||||||
import { LovelaceCardConfig } from "../../../src/data/lovelace";
|
import { LovelaceCardConfig } from "../../../src/data/lovelace";
|
||||||
@@ -26,7 +19,7 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
|
|
||||||
@property({ attribute: false }) public hass!: MockHomeAssistant;
|
@property({ attribute: false }) public hass!: MockHomeAssistant;
|
||||||
|
|
||||||
@internalProperty() private _switching?: boolean;
|
@state() private _switching?: boolean;
|
||||||
|
|
||||||
private _hidden = localStorage.hide_demo_card;
|
private _hidden = localStorage.hide_demo_card;
|
||||||
|
|
||||||
@@ -113,7 +106,7 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
css`
|
css`
|
||||||
a {
|
a {
|
||||||
|
@@ -22,9 +22,9 @@ import { mockTemplate } from "./stubs/template";
|
|||||||
import { mockTranslations } from "./stubs/translations";
|
import { mockTranslations } from "./stubs/translations";
|
||||||
|
|
||||||
class HaDemo extends HomeAssistantAppEl {
|
class HaDemo extends HomeAssistantAppEl {
|
||||||
protected async _initialize() {
|
protected async _initializeHass() {
|
||||||
const initial: Partial<MockHomeAssistant> = {
|
const initial: Partial<MockHomeAssistant> = {
|
||||||
panelUrl: (this as any).panelUrl,
|
panelUrl: (this as any)._panelUrl,
|
||||||
// Override updateHass so that the correct hass lifecycle methods are called
|
// Override updateHass so that the correct hass lifecycle methods are called
|
||||||
updateHass: (hassUpdate: Partial<HomeAssistant>) =>
|
updateHass: (hassUpdate: Partial<HomeAssistant>) =>
|
||||||
this._updateHass(hassUpdate),
|
this._updateHass(hassUpdate),
|
||||||
@@ -70,7 +70,7 @@ class HaDemo extends HomeAssistantAppEl {
|
|||||||
}
|
}
|
||||||
|
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
navigate(this, href);
|
navigate(href);
|
||||||
},
|
},
|
||||||
{ capture: true }
|
{ capture: true }
|
||||||
);
|
);
|
||||||
|
@@ -3,8 +3,6 @@ import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|||||||
export const mockTranslations = (hass: MockHomeAssistant) => {
|
export const mockTranslations = (hass: MockHomeAssistant) => {
|
||||||
hass.mockWS(
|
hass.mockWS(
|
||||||
"frontend/get_translations",
|
"frontend/get_translations",
|
||||||
(/* msg: {language: string, category: string} */) => {
|
(/* msg: {language: string, category: string} */) => ({ resources: {} })
|
||||||
return { resources: {} };
|
|
||||||
}
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
|
import { html } from "@polymer/polymer/lib/utils/html-tag";
|
||||||
/* eslint-plugin-disable lit */
|
/* eslint-plugin-disable lit */
|
||||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
|
import { PolymerElement } from "@polymer/polymer/polymer-element";
|
||||||
import { safeLoad } from "js-yaml";
|
import { load } from "js-yaml";
|
||||||
import { createCardElement } from "../../../src/panels/lovelace/create-element/create-card-element";
|
import { createCardElement } from "../../../src/panels/lovelace/create-element/create-card-element";
|
||||||
|
|
||||||
class DemoCard extends PolymerElement {
|
class DemoCard extends PolymerElement {
|
||||||
@@ -15,6 +15,10 @@ class DemoCard extends PolymerElement {
|
|||||||
margin: 0 0 20px;
|
margin: 0 0 20px;
|
||||||
color: var(--primary-color);
|
color: var(--primary-color);
|
||||||
}
|
}
|
||||||
|
h2 small {
|
||||||
|
font-size: 0.5em;
|
||||||
|
color: var(--primary-text-color);
|
||||||
|
}
|
||||||
#card {
|
#card {
|
||||||
max-width: 400px;
|
max-width: 400px;
|
||||||
width: 100vw;
|
width: 100vw;
|
||||||
@@ -34,7 +38,12 @@ class DemoCard extends PolymerElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
<h2>[[config.heading]]</h2>
|
<h2>
|
||||||
|
[[config.heading]]
|
||||||
|
<template is="dom-if" if="[[_size]]">
|
||||||
|
<small>(size [[_size]])</small>
|
||||||
|
</template>
|
||||||
|
</h2>
|
||||||
<div class="root">
|
<div class="root">
|
||||||
<div id="card"></div>
|
<div id="card"></div>
|
||||||
<template is="dom-if" if="[[showConfig]]">
|
<template is="dom-if" if="[[showConfig]]">
|
||||||
@@ -55,6 +64,9 @@ class DemoCard extends PolymerElement {
|
|||||||
observer: "_configChanged",
|
observer: "_configChanged",
|
||||||
},
|
},
|
||||||
showConfig: Boolean,
|
showConfig: Boolean,
|
||||||
|
_size: {
|
||||||
|
type: Number,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -68,8 +80,19 @@ class DemoCard extends PolymerElement {
|
|||||||
card.removeChild(card.lastChild);
|
card.removeChild(card.lastChild);
|
||||||
}
|
}
|
||||||
|
|
||||||
const el = this._createCardElement(safeLoad(config.config)[0]);
|
const el = this._createCardElement(load(config.config)[0]);
|
||||||
card.appendChild(el);
|
card.appendChild(el);
|
||||||
|
this._getSize(el);
|
||||||
|
}
|
||||||
|
|
||||||
|
async _getSize(el) {
|
||||||
|
await customElements.whenDefined(el.localName);
|
||||||
|
|
||||||
|
if (!("getCardSize" in el)) {
|
||||||
|
this._size = undefined;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this._size = await el.getCardSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
_createCardElement(cardConfig) {
|
_createCardElement(cardConfig) {
|
||||||
|
349
gallery/src/data/traces/basic_trace.ts
Normal file
349
gallery/src/data/traces/basic_trace.ts
Normal file
@@ -0,0 +1,349 @@
|
|||||||
|
import { DemoTrace } from "./types";
|
||||||
|
|
||||||
|
export const basicTrace: DemoTrace = {
|
||||||
|
trace: {
|
||||||
|
last_step: "action/2",
|
||||||
|
run_id: "0",
|
||||||
|
state: "stopped",
|
||||||
|
timestamp: {
|
||||||
|
start: "2021-03-25T04:36:51.223693+00:00",
|
||||||
|
finish: "2021-03-25T04:36:51.266132+00:00",
|
||||||
|
},
|
||||||
|
trigger: "state of input_boolean.toggle_1",
|
||||||
|
domain: "automation",
|
||||||
|
item_id: "1615419646544",
|
||||||
|
trace: {
|
||||||
|
"trigger/0": [
|
||||||
|
{
|
||||||
|
path: "trigger/0",
|
||||||
|
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"condition/0": [
|
||||||
|
{
|
||||||
|
path: "condition/0",
|
||||||
|
timestamp: "2021-03-25T04:36:51.228243+00:00",
|
||||||
|
changed_variables: {
|
||||||
|
trigger: {
|
||||||
|
platform: "state",
|
||||||
|
entity_id: "input_boolean.toggle_1",
|
||||||
|
from_state: {
|
||||||
|
entity_id: "input_boolean.toggle_1",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
editable: true,
|
||||||
|
friendly_name: "Toggle 1",
|
||||||
|
},
|
||||||
|
last_changed: "2021-03-24T19:03:59.141440+00:00",
|
||||||
|
last_updated: "2021-03-24T19:03:59.141440+00:00",
|
||||||
|
context: {
|
||||||
|
id: "5d0918eb379214d07554bdab6a08bcff",
|
||||||
|
parent_id: null,
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
to_state: {
|
||||||
|
entity_id: "input_boolean.toggle_1",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
editable: true,
|
||||||
|
friendly_name: "Toggle 1",
|
||||||
|
},
|
||||||
|
last_changed: "2021-03-25T04:36:51.220696+00:00",
|
||||||
|
last_updated: "2021-03-25T04:36:51.220696+00:00",
|
||||||
|
context: {
|
||||||
|
id: "664d6d261450a9ecea6738e97269a149",
|
||||||
|
parent_id: null,
|
||||||
|
user_id: "d1b4e89da01445fa8bc98e39fac477ca",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
for: null,
|
||||||
|
attribute: null,
|
||||||
|
description: "state of input_boolean.toggle_1",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
result: {
|
||||||
|
result: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/0": [
|
||||||
|
{
|
||||||
|
path: "action/0",
|
||||||
|
timestamp: "2021-03-25T04:36:51.243018+00:00",
|
||||||
|
changed_variables: {
|
||||||
|
trigger: {
|
||||||
|
platform: "state",
|
||||||
|
entity_id: "input_boolean.toggle_1",
|
||||||
|
from_state: {
|
||||||
|
entity_id: "input_boolean.toggle_1",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
editable: true,
|
||||||
|
friendly_name: "Toggle 1",
|
||||||
|
},
|
||||||
|
last_changed: "2021-03-24T19:03:59.141440+00:00",
|
||||||
|
last_updated: "2021-03-24T19:03:59.141440+00:00",
|
||||||
|
context: {
|
||||||
|
id: "5d0918eb379214d07554bdab6a08bcff",
|
||||||
|
parent_id: null,
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
to_state: {
|
||||||
|
entity_id: "input_boolean.toggle_1",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
editable: true,
|
||||||
|
friendly_name: "Toggle 1",
|
||||||
|
},
|
||||||
|
last_changed: "2021-03-25T04:36:51.220696+00:00",
|
||||||
|
last_updated: "2021-03-25T04:36:51.220696+00:00",
|
||||||
|
context: {
|
||||||
|
id: "664d6d261450a9ecea6738e97269a149",
|
||||||
|
parent_id: null,
|
||||||
|
user_id: "d1b4e89da01445fa8bc98e39fac477ca",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
for: null,
|
||||||
|
attribute: null,
|
||||||
|
description: "state of input_boolean.toggle_1",
|
||||||
|
},
|
||||||
|
context: {
|
||||||
|
id: "6cfcae368e7b3686fad6c59e83ae76c9",
|
||||||
|
parent_id: "664d6d261450a9ecea6738e97269a149",
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
result: {
|
||||||
|
params: {
|
||||||
|
domain: "input_boolean",
|
||||||
|
service: "toggle",
|
||||||
|
service_data: {},
|
||||||
|
target: {
|
||||||
|
entity_id: ["input_boolean.toggle_4"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
running_script: false,
|
||||||
|
limit: 10,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/1": [
|
||||||
|
{
|
||||||
|
path: "action/1",
|
||||||
|
timestamp: "2021-03-25T04:36:51.252406+00:00",
|
||||||
|
result: {
|
||||||
|
choice: 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/1/choose/0": [
|
||||||
|
{
|
||||||
|
path: "action/1/choose/0",
|
||||||
|
timestamp: "2021-03-25T04:36:51.254569+00:00",
|
||||||
|
result: {
|
||||||
|
result: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/1/choose/0/conditions/0": [
|
||||||
|
{
|
||||||
|
path: "action/1/choose/0/conditions/0",
|
||||||
|
timestamp: "2021-03-25T04:36:51.254697+00:00",
|
||||||
|
result: {
|
||||||
|
result: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/1/choose/0/sequence/0": [
|
||||||
|
{
|
||||||
|
path: "action/1/choose/0/sequence/0",
|
||||||
|
timestamp: "2021-03-25T04:36:51.257360+00:00",
|
||||||
|
result: {
|
||||||
|
params: {
|
||||||
|
domain: "input_boolean",
|
||||||
|
service: "toggle",
|
||||||
|
service_data: {},
|
||||||
|
target: {
|
||||||
|
entity_id: ["input_boolean.toggle_2"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
running_script: false,
|
||||||
|
limit: 10,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/1/choose/0/sequence/1": [
|
||||||
|
{
|
||||||
|
path: "action/1/choose/0/sequence/1",
|
||||||
|
timestamp: "2021-03-25T04:36:51.260658+00:00",
|
||||||
|
result: {
|
||||||
|
params: {
|
||||||
|
domain: "input_boolean",
|
||||||
|
service: "toggle",
|
||||||
|
service_data: {},
|
||||||
|
target: {
|
||||||
|
entity_id: ["input_boolean.toggle_3"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
running_script: false,
|
||||||
|
limit: 10,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/2": [
|
||||||
|
{
|
||||||
|
path: "action/2",
|
||||||
|
timestamp: "2021-03-25T04:36:51.264159+00:00",
|
||||||
|
result: {
|
||||||
|
params: {
|
||||||
|
domain: "input_boolean",
|
||||||
|
service: "toggle",
|
||||||
|
service_data: {},
|
||||||
|
target: {
|
||||||
|
entity_id: ["input_boolean.toggle_4"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
running_script: false,
|
||||||
|
limit: 10,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
|
||||||
|
config: {
|
||||||
|
id: "1615419646544",
|
||||||
|
alias: "Ensure Party mode",
|
||||||
|
description: "",
|
||||||
|
trigger: [
|
||||||
|
{
|
||||||
|
platform: "state",
|
||||||
|
entity_id: "input_boolean.toggle_1",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
condition: [
|
||||||
|
{
|
||||||
|
condition: "template",
|
||||||
|
alias: "Test if Paulus is home",
|
||||||
|
value_template: "{{ true }}",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
action: [
|
||||||
|
{
|
||||||
|
service: "input_boolean.toggle",
|
||||||
|
target: {
|
||||||
|
entity_id: "input_boolean.toggle_4",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
choose: [
|
||||||
|
{
|
||||||
|
alias: "If toggle 3 is on",
|
||||||
|
conditions: [
|
||||||
|
{
|
||||||
|
condition: "template",
|
||||||
|
value_template:
|
||||||
|
"{{ is_state('input_boolean.toggle_3', 'on') }}",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
sequence: [
|
||||||
|
{
|
||||||
|
service: "input_boolean.toggle",
|
||||||
|
alias: "Toggle 2 while 3 is on",
|
||||||
|
target: {
|
||||||
|
entity_id: "input_boolean.toggle_2",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
service: "input_boolean.toggle",
|
||||||
|
alias: "Toggle 3",
|
||||||
|
target: {
|
||||||
|
entity_id: "input_boolean.toggle_3",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
default: [
|
||||||
|
{
|
||||||
|
service: "input_boolean.toggle",
|
||||||
|
alias: "Toggle 2",
|
||||||
|
target: {
|
||||||
|
entity_id: "input_boolean.toggle_2",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
service: "input_boolean.toggle",
|
||||||
|
target: {
|
||||||
|
entity_id: "input_boolean.toggle_4",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
mode: "single",
|
||||||
|
},
|
||||||
|
context: {
|
||||||
|
id: "6cfcae368e7b3686fad6c59e83ae76c9",
|
||||||
|
parent_id: "664d6d261450a9ecea6738e97269a149",
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
script_execution: "finished",
|
||||||
|
},
|
||||||
|
logbookEntries: [
|
||||||
|
{
|
||||||
|
name: "Ensure Party mode",
|
||||||
|
message: "has been triggered by state of input_boolean.toggle_1",
|
||||||
|
source: "state of input_boolean.toggle_1",
|
||||||
|
entity_id: "automation.toggle_toggles",
|
||||||
|
context_id: "6cfcae368e7b3686fad6c59e83ae76c9",
|
||||||
|
when: "2021-03-25T04:36:51.240832+00:00",
|
||||||
|
domain: "automation",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
when: "2021-03-25T04:36:51.249828+00:00",
|
||||||
|
name: "Toggle 4",
|
||||||
|
state: "on",
|
||||||
|
entity_id: "input_boolean.toggle_4",
|
||||||
|
context_entity_id: "automation.toggle_toggles",
|
||||||
|
context_entity_id_name: "Ensure Party mode",
|
||||||
|
context_event_type: "automation_triggered",
|
||||||
|
context_domain: "automation",
|
||||||
|
context_name: "Ensure Party mode",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
when: "2021-03-25T04:36:51.258947+00:00",
|
||||||
|
name: "Toggle 2",
|
||||||
|
state: "on",
|
||||||
|
entity_id: "input_boolean.toggle_2",
|
||||||
|
context_entity_id: "automation.toggle_toggles",
|
||||||
|
context_entity_id_name: "Ensure Party mode",
|
||||||
|
context_event_type: "automation_triggered",
|
||||||
|
context_domain: "automation",
|
||||||
|
context_name: "Ensure Party mode",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
when: "2021-03-25T04:36:51.261806+00:00",
|
||||||
|
name: "Toggle 3",
|
||||||
|
state: "off",
|
||||||
|
entity_id: "input_boolean.toggle_3",
|
||||||
|
context_entity_id: "automation.toggle_toggles",
|
||||||
|
context_entity_id_name: "Ensure Party mode",
|
||||||
|
context_event_type: "automation_triggered",
|
||||||
|
context_domain: "automation",
|
||||||
|
context_name: "Ensure Party mode",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
when: "2021-03-25T04:36:51.265246+00:00",
|
||||||
|
name: "Toggle 4",
|
||||||
|
state: "off",
|
||||||
|
entity_id: "input_boolean.toggle_4",
|
||||||
|
context_entity_id: "automation.toggle_toggles",
|
||||||
|
context_entity_id_name: "Ensure Party mode",
|
||||||
|
context_event_type: "automation_triggered",
|
||||||
|
context_domain: "automation",
|
||||||
|
context_name: "Ensure Party mode",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
44
gallery/src/data/traces/mock-demo-trace.ts
Normal file
44
gallery/src/data/traces/mock-demo-trace.ts
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import { LogbookEntry } from "../../../../src/data/logbook";
|
||||||
|
import { AutomationTraceExtended } from "../../../../src/data/trace";
|
||||||
|
import { DemoTrace } from "./types";
|
||||||
|
|
||||||
|
export const mockDemoTrace = (
|
||||||
|
tracePartial: Partial<AutomationTraceExtended>,
|
||||||
|
logbookEntries?: LogbookEntry[]
|
||||||
|
): DemoTrace => ({
|
||||||
|
trace: {
|
||||||
|
last_step: "",
|
||||||
|
run_id: "0",
|
||||||
|
state: "stopped",
|
||||||
|
timestamp: {
|
||||||
|
start: "2021-03-25T04:36:51.223693+00:00",
|
||||||
|
finish: "2021-03-25T04:36:51.266132+00:00",
|
||||||
|
},
|
||||||
|
trigger: "mocked trigger",
|
||||||
|
domain: "automation",
|
||||||
|
item_id: "1615419646544",
|
||||||
|
trace: {
|
||||||
|
"trigger/0": [
|
||||||
|
{
|
||||||
|
path: "trigger/0",
|
||||||
|
changed_variables: {
|
||||||
|
trigger: {
|
||||||
|
description: "mocked trigger",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
config: {
|
||||||
|
trigger: [],
|
||||||
|
action: [],
|
||||||
|
},
|
||||||
|
context: {
|
||||||
|
id: "abcd",
|
||||||
|
},
|
||||||
|
script_execution: "finished",
|
||||||
|
...tracePartial,
|
||||||
|
},
|
||||||
|
logbookEntries: logbookEntries || [],
|
||||||
|
});
|
214
gallery/src/data/traces/motion-light-trace.ts
Normal file
214
gallery/src/data/traces/motion-light-trace.ts
Normal file
@@ -0,0 +1,214 @@
|
|||||||
|
import { DemoTrace } from "./types";
|
||||||
|
|
||||||
|
export const motionLightTrace: DemoTrace = {
|
||||||
|
trace: {
|
||||||
|
last_step: "action/3",
|
||||||
|
run_id: "1",
|
||||||
|
state: "stopped",
|
||||||
|
timestamp: {
|
||||||
|
start: "2021-03-14T06:07:01.768006+00:00",
|
||||||
|
finish: "2021-03-14T06:07:53.287525+00:00",
|
||||||
|
},
|
||||||
|
trigger: "state of binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
domain: "automation",
|
||||||
|
item_id: "1614732497392",
|
||||||
|
trace: {
|
||||||
|
"trigger/0": [
|
||||||
|
{
|
||||||
|
path: "trigger/0",
|
||||||
|
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/0": [
|
||||||
|
{
|
||||||
|
path: "action/0",
|
||||||
|
timestamp: "2021-03-14T06:07:01.771038+00:00",
|
||||||
|
changed_variables: {
|
||||||
|
trigger: {
|
||||||
|
platform: "state",
|
||||||
|
entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
from_state: {
|
||||||
|
entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
friendly_name: "Paulus’s MacBook Pro Camera In Use",
|
||||||
|
icon: "mdi:camera-off",
|
||||||
|
},
|
||||||
|
last_changed: "2021-03-14T06:06:29.235325+00:00",
|
||||||
|
last_updated: "2021-03-14T06:06:29.235325+00:00",
|
||||||
|
context: {
|
||||||
|
id: "ad4864c5ce957c38a07b50378eeb245d",
|
||||||
|
parent_id: null,
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
to_state: {
|
||||||
|
entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
friendly_name: "Paulus’s MacBook Pro Camera In Use",
|
||||||
|
icon: "mdi:camera",
|
||||||
|
},
|
||||||
|
last_changed: "2021-03-14T06:07:01.762009+00:00",
|
||||||
|
last_updated: "2021-03-14T06:07:01.762009+00:00",
|
||||||
|
context: {
|
||||||
|
id: "e22ddfd5f11dc4aad9a52fc10dab613b",
|
||||||
|
parent_id: null,
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
for: null,
|
||||||
|
attribute: null,
|
||||||
|
description:
|
||||||
|
"state of binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
},
|
||||||
|
context: {
|
||||||
|
id: "43b6ee9293a551c5cc14e8eb60af54ba",
|
||||||
|
parent_id: "e22ddfd5f11dc4aad9a52fc10dab613b",
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/1": [
|
||||||
|
{ path: "action/1", timestamp: "2021-03-14T06:07:01.875316+00:00" },
|
||||||
|
],
|
||||||
|
"action/2": [
|
||||||
|
{
|
||||||
|
path: "action/2",
|
||||||
|
timestamp: "2021-03-14T06:07:53.195013+00:00",
|
||||||
|
changed_variables: {
|
||||||
|
wait: {
|
||||||
|
remaining: null,
|
||||||
|
trigger: {
|
||||||
|
platform: "state",
|
||||||
|
entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
from_state: {
|
||||||
|
entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
state: "on",
|
||||||
|
attributes: {
|
||||||
|
friendly_name: "Paulus’s MacBook Pro Camera In Use",
|
||||||
|
icon: "mdi:camera",
|
||||||
|
},
|
||||||
|
last_changed: "2021-03-14T06:07:01.762009+00:00",
|
||||||
|
last_updated: "2021-03-14T06:07:01.762009+00:00",
|
||||||
|
context: {
|
||||||
|
id: "e22ddfd5f11dc4aad9a52fc10dab613b",
|
||||||
|
parent_id: null,
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
to_state: {
|
||||||
|
entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
state: "off",
|
||||||
|
attributes: {
|
||||||
|
friendly_name: "Paulus’s MacBook Pro Camera In Use",
|
||||||
|
icon: "mdi:camera-off",
|
||||||
|
},
|
||||||
|
last_changed: "2021-03-14T06:07:53.186755+00:00",
|
||||||
|
last_updated: "2021-03-14T06:07:53.186755+00:00",
|
||||||
|
context: {
|
||||||
|
id: "b2308cc91d509ea8e0c623331ab178d6",
|
||||||
|
parent_id: null,
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
for: null,
|
||||||
|
attribute: null,
|
||||||
|
description:
|
||||||
|
"state of binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"action/3": [
|
||||||
|
{
|
||||||
|
path: "action/3",
|
||||||
|
timestamp: "2021-03-14T06:07:53.196014+00:00",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
config: {
|
||||||
|
mode: "restart",
|
||||||
|
max_exceeded: "silent",
|
||||||
|
trigger: [
|
||||||
|
{
|
||||||
|
platform: "state",
|
||||||
|
entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
from: "off",
|
||||||
|
to: "on",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
action: [
|
||||||
|
{
|
||||||
|
service: "light.turn_on",
|
||||||
|
target: {
|
||||||
|
entity_id: "light.elgato_key_light_air",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
wait_for_trigger: [
|
||||||
|
{
|
||||||
|
platform: "state",
|
||||||
|
entity_id: "binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
from: "on",
|
||||||
|
to: "off",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
delay: 0,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
service: "light.turn_off",
|
||||||
|
target: {
|
||||||
|
entity_id: "light.elgato_key_light_air",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
id: "1614732497392",
|
||||||
|
alias: "Auto Elgato",
|
||||||
|
description: "",
|
||||||
|
},
|
||||||
|
context: {
|
||||||
|
id: "43b6ee9293a551c5cc14e8eb60af54ba",
|
||||||
|
parent_id: "e22ddfd5f11dc4aad9a52fc10dab613b",
|
||||||
|
user_id: null,
|
||||||
|
},
|
||||||
|
script_execution: "finished",
|
||||||
|
},
|
||||||
|
logbookEntries: [
|
||||||
|
{
|
||||||
|
name: "Auto Elgato",
|
||||||
|
message:
|
||||||
|
"has been triggered by state of binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
source: "state of binary_sensor.pauluss_macbook_pro_camera_in_use",
|
||||||
|
entity_id: "automation.auto_elgato",
|
||||||
|
when: "2021-03-14T06:07:01.768492+00:00",
|
||||||
|
domain: "automation",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
when: "2021-03-14T06:07:01.872187+00:00",
|
||||||
|
name: "Elgato Key Light Air",
|
||||||
|
state: "on",
|
||||||
|
entity_id: "light.elgato_key_light_air",
|
||||||
|
context_entity_id: "automation.auto_elgato",
|
||||||
|
context_entity_id_name: "Auto Elgato",
|
||||||
|
context_event_type: "automation_triggered",
|
||||||
|
context_domain: "automation",
|
||||||
|
context_name: "Auto Elgato",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
when: "2021-03-14T06:07:53.284505+00:00",
|
||||||
|
name: "Elgato Key Light Air",
|
||||||
|
state: "off",
|
||||||
|
entity_id: "light.elgato_key_light_air",
|
||||||
|
context_entity_id: "automation.auto_elgato",
|
||||||
|
context_entity_id_name: "Auto Elgato",
|
||||||
|
context_event_type: "automation_triggered",
|
||||||
|
context_domain: "automation",
|
||||||
|
context_name: "Auto Elgato",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
7
gallery/src/data/traces/types.ts
Normal file
7
gallery/src/data/traces/types.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { AutomationTraceExtended } from "../../../../src/data/trace";
|
||||||
|
import { LogbookEntry } from "../../../../src/data/logbook";
|
||||||
|
|
||||||
|
export interface DemoTrace {
|
||||||
|
trace: AutomationTraceExtended;
|
||||||
|
logbookEntries: LogbookEntry[];
|
||||||
|
}
|
96
gallery/src/demos/demo-automation-describe-action.ts
Normal file
96
gallery/src/demos/demo-automation-describe-action.ts
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
import { dump } from "js-yaml";
|
||||||
|
import { html, css, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement, property } from "lit/decorators";
|
||||||
|
import "../../../src/components/ha-card";
|
||||||
|
import { describeAction } from "../../../src/data/script_i18n";
|
||||||
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
|
import { HomeAssistant } from "../../../src/types";
|
||||||
|
|
||||||
|
const actions = [
|
||||||
|
{ wait_template: "{{ true }}", alias: "Something with an alias" },
|
||||||
|
{ delay: "0:05" },
|
||||||
|
{ wait_template: "{{ true }}" },
|
||||||
|
{
|
||||||
|
condition: "template",
|
||||||
|
value_template: "{{ true }}",
|
||||||
|
},
|
||||||
|
{ event: "happy_event" },
|
||||||
|
{
|
||||||
|
device_id: "abcdefgh",
|
||||||
|
domain: "plex",
|
||||||
|
entity_id: "media_player.kitchen",
|
||||||
|
},
|
||||||
|
{ scene: "scene.kitchen_morning" },
|
||||||
|
{
|
||||||
|
wait_for_trigger: [
|
||||||
|
{
|
||||||
|
platform: "state",
|
||||||
|
entity_id: "input_boolean.toggle_1",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
variables: {
|
||||||
|
hello: "world",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
service: "input_boolean.toggle",
|
||||||
|
target: {
|
||||||
|
entity_id: "input_boolean.toggle_4",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
@customElement("demo-automation-describe-action")
|
||||||
|
export class DemoAutomationDescribeAction extends LitElement {
|
||||||
|
@property({ attribute: false }) hass!: HomeAssistant;
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this.hass) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
return html`
|
||||||
|
<ha-card header="Actions">
|
||||||
|
${actions.map(
|
||||||
|
(conf) => html`
|
||||||
|
<div class="action">
|
||||||
|
<span>${describeAction(this.hass, conf as any)}</span>
|
||||||
|
<pre>${dump(conf)}</pre>
|
||||||
|
</div>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</ha-card>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected firstUpdated(changedProps) {
|
||||||
|
super.firstUpdated(changedProps);
|
||||||
|
const hass = provideHass(this);
|
||||||
|
hass.updateTranslations(null, "en");
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles() {
|
||||||
|
return css`
|
||||||
|
ha-card {
|
||||||
|
max-width: 600px;
|
||||||
|
margin: 24px auto;
|
||||||
|
}
|
||||||
|
.action {
|
||||||
|
padding: 16px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
span {
|
||||||
|
margin-right: 16px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-automation-describe-action": DemoAutomationDescribeAction;
|
||||||
|
}
|
||||||
|
}
|
60
gallery/src/demos/demo-automation-describe-condition.ts
Normal file
60
gallery/src/demos/demo-automation-describe-condition.ts
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
import { dump } from "js-yaml";
|
||||||
|
import { html, css, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement } from "lit/decorators";
|
||||||
|
import "../../../src/components/ha-card";
|
||||||
|
import { describeCondition } from "../../../src/data/automation_i18n";
|
||||||
|
|
||||||
|
const conditions = [
|
||||||
|
{ condition: "and" },
|
||||||
|
{ condition: "not" },
|
||||||
|
{ condition: "or" },
|
||||||
|
{ condition: "state" },
|
||||||
|
{ condition: "numeric_state" },
|
||||||
|
{ condition: "sun", after: "sunset" },
|
||||||
|
{ condition: "sun", after: "sunrise" },
|
||||||
|
{ condition: "zone" },
|
||||||
|
{ condition: "time" },
|
||||||
|
{ condition: "template" },
|
||||||
|
];
|
||||||
|
|
||||||
|
@customElement("demo-automation-describe-condition")
|
||||||
|
export class DemoAutomationDescribeCondition extends LitElement {
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<ha-card header="Conditions">
|
||||||
|
${conditions.map(
|
||||||
|
(conf) => html`
|
||||||
|
<div class="condition">
|
||||||
|
<span>${describeCondition(conf as any)}</span>
|
||||||
|
<pre>${dump(conf)}</pre>
|
||||||
|
</div>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</ha-card>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles() {
|
||||||
|
return css`
|
||||||
|
ha-card {
|
||||||
|
max-width: 600px;
|
||||||
|
margin: 24px auto;
|
||||||
|
}
|
||||||
|
.condition {
|
||||||
|
padding: 16px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
span {
|
||||||
|
margin-right: 16px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-automation-describe-condition": DemoAutomationDescribeCondition;
|
||||||
|
}
|
||||||
|
}
|
63
gallery/src/demos/demo-automation-describe-trigger.ts
Normal file
63
gallery/src/demos/demo-automation-describe-trigger.ts
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
import { dump } from "js-yaml";
|
||||||
|
import { html, css, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement } from "lit/decorators";
|
||||||
|
import "../../../src/components/ha-card";
|
||||||
|
import { describeTrigger } from "../../../src/data/automation_i18n";
|
||||||
|
|
||||||
|
const triggers = [
|
||||||
|
{ platform: "state" },
|
||||||
|
{ platform: "mqtt" },
|
||||||
|
{ platform: "geo_location" },
|
||||||
|
{ platform: "homeassistant" },
|
||||||
|
{ platform: "numeric_state" },
|
||||||
|
{ platform: "sun" },
|
||||||
|
{ platform: "time_pattern" },
|
||||||
|
{ platform: "webhook" },
|
||||||
|
{ platform: "zone" },
|
||||||
|
{ platform: "tag" },
|
||||||
|
{ platform: "time" },
|
||||||
|
{ platform: "template" },
|
||||||
|
{ platform: "event" },
|
||||||
|
];
|
||||||
|
|
||||||
|
@customElement("demo-automation-describe-trigger")
|
||||||
|
export class DemoAutomationDescribeTrigger extends LitElement {
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<ha-card header="Triggers">
|
||||||
|
${triggers.map(
|
||||||
|
(conf) => html`
|
||||||
|
<div class="trigger">
|
||||||
|
<span>${describeTrigger(conf as any)}</span>
|
||||||
|
<pre>${dump(conf)}</pre>
|
||||||
|
</div>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</ha-card>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles() {
|
||||||
|
return css`
|
||||||
|
ha-card {
|
||||||
|
max-width: 600px;
|
||||||
|
margin: 24px auto;
|
||||||
|
}
|
||||||
|
.trigger {
|
||||||
|
padding: 16px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
span {
|
||||||
|
margin-right: 16px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-automation-describe-trigger": DemoAutomationDescribeTrigger;
|
||||||
|
}
|
||||||
|
}
|
81
gallery/src/demos/demo-automation-trace-timeline.ts
Normal file
81
gallery/src/demos/demo-automation-trace-timeline.ts
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
import { html, css, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement, property } from "lit/decorators";
|
||||||
|
import "../../../src/components/ha-card";
|
||||||
|
import "../../../src/components/trace/hat-script-graph";
|
||||||
|
import "../../../src/components/trace/hat-trace-timeline";
|
||||||
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
|
import { HomeAssistant } from "../../../src/types";
|
||||||
|
import { mockDemoTrace } from "../data/traces/mock-demo-trace";
|
||||||
|
import { DemoTrace } from "../data/traces/types";
|
||||||
|
|
||||||
|
const traces: DemoTrace[] = [
|
||||||
|
mockDemoTrace({ state: "running" }),
|
||||||
|
mockDemoTrace({ state: "debugged" }),
|
||||||
|
mockDemoTrace({ state: "stopped", script_execution: "failed_conditions" }),
|
||||||
|
mockDemoTrace({ state: "stopped", script_execution: "failed_single" }),
|
||||||
|
mockDemoTrace({ state: "stopped", script_execution: "failed_max_runs" }),
|
||||||
|
mockDemoTrace({ state: "stopped", script_execution: "finished" }),
|
||||||
|
mockDemoTrace({ state: "stopped", script_execution: "aborted" }),
|
||||||
|
mockDemoTrace({
|
||||||
|
state: "stopped",
|
||||||
|
script_execution: "error",
|
||||||
|
error: 'Variable "beer" cannot be None',
|
||||||
|
}),
|
||||||
|
mockDemoTrace({ state: "stopped", script_execution: "cancelled" }),
|
||||||
|
];
|
||||||
|
|
||||||
|
@customElement("demo-automation-trace-timeline")
|
||||||
|
export class DemoAutomationTraceTimeline extends LitElement {
|
||||||
|
@property({ attribute: false }) hass?: HomeAssistant;
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this.hass) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
return html`
|
||||||
|
${traces.map(
|
||||||
|
(trace) => html`
|
||||||
|
<ha-card .header=${trace.trace.config.alias}>
|
||||||
|
<div class="card-content">
|
||||||
|
<hat-trace-timeline
|
||||||
|
.hass=${this.hass}
|
||||||
|
.trace=${trace.trace}
|
||||||
|
.logbookEntries=${trace.logbookEntries}
|
||||||
|
></hat-trace-timeline>
|
||||||
|
<button @click=${() => console.log(trace)}>Log trace</button>
|
||||||
|
</div>
|
||||||
|
</ha-card>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected firstUpdated(changedProps) {
|
||||||
|
super.firstUpdated(changedProps);
|
||||||
|
const hass = provideHass(this);
|
||||||
|
hass.updateTranslations(null, "en");
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles() {
|
||||||
|
return css`
|
||||||
|
ha-card {
|
||||||
|
max-width: 600px;
|
||||||
|
margin: 24px;
|
||||||
|
}
|
||||||
|
.card-content {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
button {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-automation-trace-timeline": DemoAutomationTraceTimeline;
|
||||||
|
}
|
||||||
|
}
|
91
gallery/src/demos/demo-automation-trace.ts
Normal file
91
gallery/src/demos/demo-automation-trace.ts
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
import { html, css, LitElement, TemplateResult } from "lit";
|
||||||
|
import "../../../src/components/ha-card";
|
||||||
|
import "../../../src/components/trace/hat-script-graph";
|
||||||
|
import "../../../src/components/trace/hat-trace-timeline";
|
||||||
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
|
import { HomeAssistant } from "../../../src/types";
|
||||||
|
import { DemoTrace } from "../data/traces/types";
|
||||||
|
import { basicTrace } from "../data/traces/basic_trace";
|
||||||
|
import { motionLightTrace } from "../data/traces/motion-light-trace";
|
||||||
|
import { customElement, property, state } from "lit/decorators";
|
||||||
|
|
||||||
|
const traces: DemoTrace[] = [basicTrace, motionLightTrace];
|
||||||
|
|
||||||
|
@customElement("demo-automation-trace")
|
||||||
|
export class DemoAutomationTrace extends LitElement {
|
||||||
|
@property({ attribute: false }) hass?: HomeAssistant;
|
||||||
|
|
||||||
|
@state() private _selected = {};
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this.hass) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
return html`
|
||||||
|
${traces.map(
|
||||||
|
(trace, idx) => html`
|
||||||
|
<ha-card .header=${trace.trace.config.alias}>
|
||||||
|
<div class="card-content">
|
||||||
|
<hat-script-graph
|
||||||
|
.trace=${trace.trace}
|
||||||
|
.selected=${this._selected[idx]}
|
||||||
|
@graph-node-selected=${(ev) => {
|
||||||
|
this._selected = { ...this._selected, [idx]: ev.detail.path };
|
||||||
|
}}
|
||||||
|
></hat-script-graph>
|
||||||
|
<hat-trace-timeline
|
||||||
|
allowPick
|
||||||
|
.hass=${this.hass}
|
||||||
|
.trace=${trace.trace}
|
||||||
|
.logbookEntries=${trace.logbookEntries}
|
||||||
|
.selectedPath=${this._selected[idx]}
|
||||||
|
@value-changed=${(ev) => {
|
||||||
|
this._selected = {
|
||||||
|
...this._selected,
|
||||||
|
[idx]: ev.detail.value,
|
||||||
|
};
|
||||||
|
}}
|
||||||
|
></hat-trace-timeline>
|
||||||
|
<button @click=${() => console.log(trace)}>Log trace</button>
|
||||||
|
</div>
|
||||||
|
</ha-card>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected firstUpdated(changedProps) {
|
||||||
|
super.firstUpdated(changedProps);
|
||||||
|
const hass = provideHass(this);
|
||||||
|
hass.updateTranslations(null, "en");
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles() {
|
||||||
|
return css`
|
||||||
|
ha-card {
|
||||||
|
max-width: 600px;
|
||||||
|
margin: 24px;
|
||||||
|
}
|
||||||
|
.card-content {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
.card-content > * {
|
||||||
|
margin-right: 16px;
|
||||||
|
}
|
||||||
|
.card-content > *:last-child {
|
||||||
|
margin-right: 0;
|
||||||
|
}
|
||||||
|
button {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
right: 0;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-automation-trace": DemoAutomationTrace;
|
||||||
|
}
|
||||||
|
}
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -93,4 +87,8 @@ class DemoAlarmPanelEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-alarm-panel-card", DemoAlarmPanelEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-alarm-panel-card": DemoAlarmPanelEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -75,4 +69,8 @@ class DemoConditional extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-conditional-card", DemoConditional);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-conditional-card": DemoConditional;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -239,4 +233,8 @@ class DemoEntities extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-entities-card", DemoEntities);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-entities-card": DemoEntities;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -91,4 +85,8 @@ class DemoButtonEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-entity-button-card", DemoButtonEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-entity-button-card": DemoButtonEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -132,4 +126,8 @@ class DemoEntityFilter extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-entity-filter-card", DemoEntityFilter);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-entity-filter-card": DemoEntityFilter;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -129,4 +123,8 @@ class DemoGaugeEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-gauge-card", DemoGaugeEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-gauge-card": DemoGaugeEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -232,4 +226,8 @@ class DemoGlanceEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-glance-card", DemoGlanceEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-glance-card": DemoGlanceEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { mockHistory } from "../../../demo/src/stubs/history";
|
import { mockHistory } from "../../../demo/src/stubs/history";
|
||||||
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";
|
||||||
@@ -49,6 +43,110 @@ const ENTITIES = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
const CONFIGS = [
|
const CONFIGS = [
|
||||||
|
{
|
||||||
|
heading: "Default Grid",
|
||||||
|
config: `
|
||||||
|
- type: grid
|
||||||
|
cards:
|
||||||
|
- type: entity
|
||||||
|
entity: light.kitchen_lights
|
||||||
|
- type: entity
|
||||||
|
entity: light.bed_light
|
||||||
|
- type: entity
|
||||||
|
entity: device_tracker.demo_paulus
|
||||||
|
- type: sensor
|
||||||
|
entity: sensor.illumination
|
||||||
|
graph: line
|
||||||
|
- type: entity
|
||||||
|
entity: device_tracker.demo_anne_therese
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Non-square Grid with 2 columns",
|
||||||
|
config: `
|
||||||
|
- type: grid
|
||||||
|
columns: 2
|
||||||
|
square: false
|
||||||
|
cards:
|
||||||
|
- type: entity
|
||||||
|
entity: light.kitchen_lights
|
||||||
|
- type: entity
|
||||||
|
entity: light.bed_light
|
||||||
|
- type: entity
|
||||||
|
entity: device_tracker.demo_paulus
|
||||||
|
- type: sensor
|
||||||
|
entity: sensor.illumination
|
||||||
|
graph: line
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Default Grid with title",
|
||||||
|
config: `
|
||||||
|
- type: grid
|
||||||
|
title: Kitchen
|
||||||
|
cards:
|
||||||
|
- type: entity
|
||||||
|
entity: light.kitchen_lights
|
||||||
|
- type: entity
|
||||||
|
entity: light.bed_light
|
||||||
|
- type: entity
|
||||||
|
entity: device_tracker.demo_paulus
|
||||||
|
- type: sensor
|
||||||
|
entity: sensor.illumination
|
||||||
|
graph: line
|
||||||
|
- type: entity
|
||||||
|
entity: device_tracker.demo_anne_therese
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Columns 4",
|
||||||
|
config: `
|
||||||
|
- type: grid
|
||||||
|
columns: 4
|
||||||
|
cards:
|
||||||
|
- type: entity
|
||||||
|
entity: light.kitchen_lights
|
||||||
|
- type: entity
|
||||||
|
entity: light.bed_light
|
||||||
|
- type: entity
|
||||||
|
entity: device_tracker.demo_paulus
|
||||||
|
- type: sensor
|
||||||
|
entity: sensor.illumination
|
||||||
|
graph: line
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Columns 2",
|
||||||
|
config: `
|
||||||
|
- type: grid
|
||||||
|
columns: 2
|
||||||
|
cards:
|
||||||
|
- type: entity
|
||||||
|
entity: light.kitchen_lights
|
||||||
|
- type: entity
|
||||||
|
entity: light.bed_light
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Columns 1",
|
||||||
|
config: `
|
||||||
|
- type: grid
|
||||||
|
columns: 1
|
||||||
|
cards:
|
||||||
|
- type: entity
|
||||||
|
entity: light.kitchen_lights
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Size for single card",
|
||||||
|
config: `
|
||||||
|
- type: grid
|
||||||
|
cards:
|
||||||
|
- type: entity
|
||||||
|
entity: light.kitchen_lights
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
heading: "Vertical Stack",
|
heading: "Vertical Stack",
|
||||||
config: `
|
config: `
|
||||||
@@ -99,45 +197,9 @@ const CONFIGS = [
|
|||||||
entity: light.bed_light
|
entity: light.bed_light
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
heading: "Default Grid",
|
|
||||||
config: `
|
|
||||||
- type: grid
|
|
||||||
cards:
|
|
||||||
- type: entity
|
|
||||||
entity: light.kitchen_lights
|
|
||||||
- type: entity
|
|
||||||
entity: light.bed_light
|
|
||||||
- type: entity
|
|
||||||
entity: device_tracker.demo_paulus
|
|
||||||
- type: sensor
|
|
||||||
entity: sensor.illumination
|
|
||||||
graph: line
|
|
||||||
- type: entity
|
|
||||||
entity: device_tracker.demo_anne_therese
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
heading: "Non-square Grid with 2 columns",
|
|
||||||
config: `
|
|
||||||
- type: grid
|
|
||||||
columns: 2
|
|
||||||
square: false
|
|
||||||
cards:
|
|
||||||
- type: entity
|
|
||||||
entity: light.kitchen_lights
|
|
||||||
- type: entity
|
|
||||||
entity: light.bed_light
|
|
||||||
- type: entity
|
|
||||||
entity: device_tracker.demo_paulus
|
|
||||||
- type: sensor
|
|
||||||
entity: sensor.illumination
|
|
||||||
graph: line
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
|
||||||
@customElement("demo-hui-stack-card")
|
@customElement("demo-hui-grid-and-stack-card")
|
||||||
class DemoStack extends LitElement {
|
class DemoStack extends LitElement {
|
||||||
@query("#demos") private _demoRoot!: HTMLElement;
|
@query("#demos") private _demoRoot!: HTMLElement;
|
||||||
|
|
||||||
@@ -155,4 +217,8 @@ class DemoStack extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-stack-card", DemoStack);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-grid-and-stack-card": DemoStack;
|
||||||
|
}
|
||||||
|
}
|
@@ -1,4 +1,5 @@
|
|||||||
import { customElement, html, LitElement, TemplateResult } from "lit-element";
|
import { html, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement } from "lit/decorators";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
|
|
||||||
const CONFIGS = [
|
const CONFIGS = [
|
||||||
@@ -42,4 +43,8 @@ class DemoIframe extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-iframe-card", DemoIframe);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-iframe-card": DemoIframe;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -85,4 +79,8 @@ class DemoLightEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-light-card", DemoLightEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-light-card": DemoLightEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -183,4 +177,8 @@ class DemoMap extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-map-card", DemoMap);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-map-card": DemoMap;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { mockTemplate } from "../../../demo/src/stubs/template";
|
import { mockTemplate } from "../../../demo/src/stubs/template";
|
||||||
import { provideHass } from "../../../src/fake_data/provide_hass";
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -276,4 +270,8 @@ class DemoMarkdown extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-markdown-card", DemoMarkdown);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-markdown-card": DemoMarkdown;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { provideHass } from "../../../src/fake_data/provide_hass";
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
import { createMediaPlayerEntities } from "../data/media_players";
|
import { createMediaPlayerEntities } from "../data/media_players";
|
||||||
@@ -180,4 +174,8 @@ class DemoHuiMediaControlCard extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-media-control-card", DemoHuiMediaControlCard);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-media-control-card": DemoHuiMediaControlCard;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { provideHass } from "../../../src/fake_data/provide_hass";
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
import { createMediaPlayerEntities } from "../data/media_players";
|
import { createMediaPlayerEntities } from "../data/media_players";
|
||||||
@@ -77,4 +71,8 @@ class DemoHuiMediaPlayerRow extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-media-player-row", DemoHuiMediaPlayerRow);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-media-player-row": DemoHuiMediaPlayerRow;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -147,4 +141,8 @@ class DemoPictureElements extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-picture-elements-card", DemoPictureElements);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-picture-elements-card": DemoPictureElements;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -102,4 +96,8 @@ class DemoPictureEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-picture-entity-card", DemoPictureEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-picture-entity-card": DemoPictureEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -143,4 +137,8 @@ class DemoPictureGlance extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-picture-glance-card", DemoPictureGlance);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-picture-glance-card": DemoPictureGlance;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { provideHass } from "../../../src/fake_data/provide_hass";
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
import { createPlantEntities } from "../data/plants";
|
import { createPlantEntities } from "../data/plants";
|
||||||
@@ -52,4 +46,8 @@ export class DemoPlantEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-plant-card", DemoPlantEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-plant-card": DemoPlantEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { provideHass } from "../../../src/fake_data/provide_hass";
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
|
|
||||||
@@ -48,4 +42,8 @@ class DemoShoppingListEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-shopping-list-card", DemoShoppingListEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-shopping-list-card": DemoShoppingListEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,11 +1,5 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
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";
|
||||||
import "../components/demo-cards";
|
import "../components/demo-cards";
|
||||||
@@ -96,4 +90,8 @@ class DemoThermostatEntity extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-hui-thermostat-card", DemoThermostatEntity);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-hui-thermostat-card": DemoThermostatEntity;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
354
gallery/src/demos/demo-integration-card.ts
Normal file
354
gallery/src/demos/demo-integration-card.ts
Normal file
@@ -0,0 +1,354 @@
|
|||||||
|
import { html, css, LitElement, TemplateResult } from "lit";
|
||||||
|
import "../../../src/components/ha-formfield";
|
||||||
|
import "../../../src/components/ha-switch";
|
||||||
|
|
||||||
|
import { IntegrationManifest } from "../../../src/data/integration";
|
||||||
|
|
||||||
|
import { provideHass } from "../../../src/fake_data/provide_hass";
|
||||||
|
import { HomeAssistant } from "../../../src/types";
|
||||||
|
import "../../../src/panels/config/integrations/ha-integration-card";
|
||||||
|
import "../../../src/panels/config/integrations/ha-ignored-config-entry-card";
|
||||||
|
import "../../../src/panels/config/integrations/ha-config-flow-card";
|
||||||
|
import type {
|
||||||
|
ConfigEntryExtended,
|
||||||
|
DataEntryFlowProgressExtended,
|
||||||
|
} from "../../../src/panels/config/integrations/ha-config-integrations";
|
||||||
|
import { DeviceRegistryEntry } from "../../../src/data/device_registry";
|
||||||
|
import { EntityRegistryEntry } from "../../../src/data/entity_registry";
|
||||||
|
import { classMap } from "lit/directives/class-map";
|
||||||
|
import { customElement, property, state } from "lit/decorators";
|
||||||
|
|
||||||
|
const createConfigEntry = (
|
||||||
|
title: string,
|
||||||
|
override: Partial<ConfigEntryExtended> = {}
|
||||||
|
): ConfigEntryExtended => ({
|
||||||
|
entry_id: title,
|
||||||
|
domain: "esphome",
|
||||||
|
localized_domain_name: "ESPHome",
|
||||||
|
title,
|
||||||
|
source: "zeroconf",
|
||||||
|
state: "loaded",
|
||||||
|
supports_options: false,
|
||||||
|
supports_unload: true,
|
||||||
|
disabled_by: null,
|
||||||
|
pref_disable_new_entities: false,
|
||||||
|
pref_disable_polling: false,
|
||||||
|
reason: null,
|
||||||
|
...override,
|
||||||
|
});
|
||||||
|
|
||||||
|
const createManifest = (
|
||||||
|
isCustom: boolean,
|
||||||
|
isCloud: boolean,
|
||||||
|
name = "ESPHome"
|
||||||
|
): IntegrationManifest => ({
|
||||||
|
name,
|
||||||
|
domain: "esphome",
|
||||||
|
is_built_in: !isCustom,
|
||||||
|
config_flow: false,
|
||||||
|
documentation: "https://www.home-assistant.io/integrations/esphome/",
|
||||||
|
iot_class: isCloud ? "cloud_polling" : "local_polling",
|
||||||
|
});
|
||||||
|
|
||||||
|
const loadedEntry = createConfigEntry("Loaded");
|
||||||
|
const nameAsDomainEntry = createConfigEntry("ESPHome");
|
||||||
|
const longNameEntry = createConfigEntry(
|
||||||
|
"Entry with a super long name that is going to the next line"
|
||||||
|
);
|
||||||
|
const longNonBreakingNameEntry = createConfigEntry(
|
||||||
|
"EntryWithASuperLongNameThatDoesNotBreak"
|
||||||
|
);
|
||||||
|
const configPanelEntry = createConfigEntry("Config Panel", {
|
||||||
|
domain: "mqtt",
|
||||||
|
localized_domain_name: "MQTT",
|
||||||
|
});
|
||||||
|
const optionsFlowEntry = createConfigEntry("Options Flow", {
|
||||||
|
supports_options: true,
|
||||||
|
});
|
||||||
|
const disabledPollingEntry = createConfigEntry("Disabled Polling", {
|
||||||
|
pref_disable_polling: true,
|
||||||
|
});
|
||||||
|
const setupErrorEntry = createConfigEntry("Setup Error", {
|
||||||
|
state: "setup_error",
|
||||||
|
});
|
||||||
|
const migrationErrorEntry = createConfigEntry("Migration Error", {
|
||||||
|
state: "migration_error",
|
||||||
|
});
|
||||||
|
const setupRetryEntry = createConfigEntry("Setup Retry", {
|
||||||
|
state: "setup_retry",
|
||||||
|
});
|
||||||
|
const setupRetryReasonEntry = createConfigEntry("Setup Retry", {
|
||||||
|
state: "setup_retry",
|
||||||
|
reason: "connection_error",
|
||||||
|
});
|
||||||
|
const setupRetryReasonMissingKeyEntry = createConfigEntry("Setup Retry", {
|
||||||
|
state: "setup_retry",
|
||||||
|
reason:
|
||||||
|
"HTTPSConnectionpool: Max retries exceeded with NewConnectionError('<urllib3.connection.HTTPSConnection object at 0x9eedfc10>: Failed to establish a new connection: [Errno 113] Host is unreachable')",
|
||||||
|
});
|
||||||
|
const failedUnloadEntry = createConfigEntry("Failed Unload", {
|
||||||
|
state: "failed_unload",
|
||||||
|
});
|
||||||
|
const notLoadedEntry = createConfigEntry("Not Loaded", { state: "not_loaded" });
|
||||||
|
const disabledEntry = createConfigEntry("Disabled", {
|
||||||
|
state: "not_loaded",
|
||||||
|
disabled_by: "user",
|
||||||
|
});
|
||||||
|
const disabledFailedUnloadEntry = createConfigEntry(
|
||||||
|
"Disabled - Failed Unload",
|
||||||
|
{
|
||||||
|
state: "failed_unload",
|
||||||
|
disabled_by: "user",
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const configFlows: DataEntryFlowProgressExtended[] = [
|
||||||
|
{
|
||||||
|
flow_id: "adbb401329d8439ebb78ef29837826a8",
|
||||||
|
handler: "roku",
|
||||||
|
context: {
|
||||||
|
source: "ssdp",
|
||||||
|
unique_id: "YF008D862864",
|
||||||
|
title_placeholders: {
|
||||||
|
name: "Living room Roku",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
step_id: "discovery_confirm",
|
||||||
|
localized_title: "Living room Roku",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
flow_id: "adbb401329d8439ebb78ef29837826a8",
|
||||||
|
handler: "hue",
|
||||||
|
context: {
|
||||||
|
source: "reauth",
|
||||||
|
unique_id: "YF008D862864",
|
||||||
|
title_placeholders: {
|
||||||
|
name: "Living room Roku",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
step_id: "discovery_confirm",
|
||||||
|
localized_title: "Philips Hue",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const configEntries: Array<{
|
||||||
|
items: ConfigEntryExtended[];
|
||||||
|
is_custom?: boolean;
|
||||||
|
disabled?: boolean;
|
||||||
|
highlight?: string;
|
||||||
|
}> = [
|
||||||
|
{ items: [loadedEntry] },
|
||||||
|
{ items: [configPanelEntry] },
|
||||||
|
{ items: [optionsFlowEntry] },
|
||||||
|
{ items: [disabledPollingEntry] },
|
||||||
|
{ items: [nameAsDomainEntry] },
|
||||||
|
{ items: [longNameEntry] },
|
||||||
|
{ items: [longNonBreakingNameEntry] },
|
||||||
|
{ items: [setupErrorEntry] },
|
||||||
|
{ items: [migrationErrorEntry] },
|
||||||
|
{ items: [setupRetryEntry] },
|
||||||
|
{ items: [setupRetryReasonEntry] },
|
||||||
|
{ items: [setupRetryReasonMissingKeyEntry] },
|
||||||
|
{ items: [failedUnloadEntry] },
|
||||||
|
{ items: [notLoadedEntry] },
|
||||||
|
{
|
||||||
|
items: [
|
||||||
|
loadedEntry,
|
||||||
|
setupErrorEntry,
|
||||||
|
migrationErrorEntry,
|
||||||
|
longNameEntry,
|
||||||
|
longNonBreakingNameEntry,
|
||||||
|
setupRetryEntry,
|
||||||
|
failedUnloadEntry,
|
||||||
|
notLoadedEntry,
|
||||||
|
disabledEntry,
|
||||||
|
nameAsDomainEntry,
|
||||||
|
configPanelEntry,
|
||||||
|
optionsFlowEntry,
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{ disabled: true, items: [disabledEntry] },
|
||||||
|
{ disabled: true, items: [disabledFailedUnloadEntry] },
|
||||||
|
{
|
||||||
|
disabled: true,
|
||||||
|
items: [disabledEntry, disabledFailedUnloadEntry],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
items: [loadedEntry, configPanelEntry],
|
||||||
|
highlight: "Loaded",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const createEntityRegistryEntries = (
|
||||||
|
item: ConfigEntryExtended
|
||||||
|
): EntityRegistryEntry[] => [
|
||||||
|
{
|
||||||
|
config_entry_id: item.entry_id,
|
||||||
|
device_id: "mock-device-id",
|
||||||
|
area_id: null,
|
||||||
|
disabled_by: null,
|
||||||
|
entity_id: "binary_sensor.updater",
|
||||||
|
name: null,
|
||||||
|
icon: null,
|
||||||
|
platform: "updater",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const createDeviceRegistryEntries = (
|
||||||
|
item: ConfigEntryExtended
|
||||||
|
): DeviceRegistryEntry[] => [
|
||||||
|
{
|
||||||
|
entry_type: null,
|
||||||
|
config_entries: [item.entry_id],
|
||||||
|
connections: [],
|
||||||
|
manufacturer: "ESPHome",
|
||||||
|
model: "Mock Device",
|
||||||
|
name: "Tag Reader",
|
||||||
|
sw_version: null,
|
||||||
|
id: "mock-device-id",
|
||||||
|
identifiers: [],
|
||||||
|
via_device_id: null,
|
||||||
|
area_id: null,
|
||||||
|
name_by_user: null,
|
||||||
|
disabled_by: null,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
@customElement("demo-integration-card")
|
||||||
|
export class DemoIntegrationCard extends LitElement {
|
||||||
|
@property({ attribute: false }) hass?: HomeAssistant;
|
||||||
|
|
||||||
|
@state() isCustomIntegration = false;
|
||||||
|
|
||||||
|
@state() isCloud = false;
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this.hass) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
return html`
|
||||||
|
<div class="container">
|
||||||
|
<div class="filters">
|
||||||
|
<ha-formfield label="Custom Integration">
|
||||||
|
<ha-switch @change=${this._toggleCustomIntegration}></ha-switch>
|
||||||
|
</ha-formfield>
|
||||||
|
<ha-formfield label="Relies on cloud">
|
||||||
|
<ha-switch @change=${this._toggleCloud}></ha-switch>
|
||||||
|
</ha-formfield>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ha-ignored-config-entry-card
|
||||||
|
.hass=${this.hass}
|
||||||
|
.entry=${createConfigEntry("Ignored Entry")}
|
||||||
|
.manifest=${createManifest(this.isCustomIntegration, this.isCloud)}
|
||||||
|
></ha-ignored-config-entry-card>
|
||||||
|
|
||||||
|
${configFlows.map(
|
||||||
|
(flow) => html`
|
||||||
|
<ha-config-flow-card
|
||||||
|
.hass=${this.hass}
|
||||||
|
.flow=${flow}
|
||||||
|
.manifest=${createManifest(
|
||||||
|
this.isCustomIntegration,
|
||||||
|
this.isCloud,
|
||||||
|
flow.handler === "roku" ? "Roku" : "Philips Hue"
|
||||||
|
)}
|
||||||
|
></ha-config-flow-card>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
${configEntries.map(
|
||||||
|
(info) => html`
|
||||||
|
<ha-integration-card
|
||||||
|
class=${classMap({
|
||||||
|
highlight: info.highlight !== undefined,
|
||||||
|
})}
|
||||||
|
.hass=${this.hass}
|
||||||
|
domain="esphome"
|
||||||
|
.items=${info.items}
|
||||||
|
.manifest=${createManifest(
|
||||||
|
this.isCustomIntegration,
|
||||||
|
this.isCloud
|
||||||
|
)}
|
||||||
|
.entityRegistryEntries=${createEntityRegistryEntries(
|
||||||
|
info.items[0]
|
||||||
|
)}
|
||||||
|
.deviceRegistryEntries=${createDeviceRegistryEntries(
|
||||||
|
info.items[0]
|
||||||
|
)}
|
||||||
|
?disabled=${info.disabled}
|
||||||
|
.selectedConfigEntryId=${info.highlight}
|
||||||
|
></ha-integration-card>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div class="container">
|
||||||
|
<!-- One that is standalone to see how it increases height if height
|
||||||
|
not defined by other cards. -->
|
||||||
|
<ha-integration-card
|
||||||
|
.hass=${this.hass}
|
||||||
|
domain="esphome"
|
||||||
|
.items=${[
|
||||||
|
loadedEntry,
|
||||||
|
setupErrorEntry,
|
||||||
|
migrationErrorEntry,
|
||||||
|
setupRetryEntry,
|
||||||
|
failedUnloadEntry,
|
||||||
|
]}
|
||||||
|
.manifest=${createManifest(this.isCustomIntegration, this.isCloud)}
|
||||||
|
.entityRegistryEntries=${createEntityRegistryEntries(loadedEntry)}
|
||||||
|
.deviceRegistryEntries=${createDeviceRegistryEntries(loadedEntry)}
|
||||||
|
></ha-integration-card>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected firstUpdated(changedProps) {
|
||||||
|
super.firstUpdated(changedProps);
|
||||||
|
const hass = provideHass(this);
|
||||||
|
hass.updateTranslations(null, "en");
|
||||||
|
hass.updateTranslations("config", "en");
|
||||||
|
// Normally this string is loaded from backend
|
||||||
|
hass.addTranslations(
|
||||||
|
{
|
||||||
|
"component.esphome.config.error.connection_error":
|
||||||
|
"Can't connect to ESP. Please make sure your YAML file contains an 'api:' line.",
|
||||||
|
},
|
||||||
|
"en"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _toggleCustomIntegration() {
|
||||||
|
this.isCustomIntegration = !this.isCustomIntegration;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _toggleCloud() {
|
||||||
|
this.isCloud = !this.isCloud;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles() {
|
||||||
|
return css`
|
||||||
|
.container {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
||||||
|
grid-gap: 16px 16px;
|
||||||
|
padding: 8px 16px 16px;
|
||||||
|
margin-bottom: 64px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container > * {
|
||||||
|
max-width: 500px;
|
||||||
|
}
|
||||||
|
|
||||||
|
ha-formfield {
|
||||||
|
margin: 8px 0;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-integration-card": DemoIntegrationCard;
|
||||||
|
}
|
||||||
|
}
|
@@ -1,21 +1,11 @@
|
|||||||
import {
|
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, property, query } from "lit/decorators";
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
PropertyValues,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../src/components/ha-card";
|
import "../../../src/components/ha-card";
|
||||||
import {
|
import {
|
||||||
SUPPORT_BRIGHTNESS,
|
LightColorModes,
|
||||||
SUPPORT_COLOR,
|
|
||||||
SUPPORT_COLOR_TEMP,
|
|
||||||
SUPPORT_EFFECT,
|
SUPPORT_EFFECT,
|
||||||
SUPPORT_FLASH,
|
SUPPORT_FLASH,
|
||||||
SUPPORT_TRANSITION,
|
SUPPORT_TRANSITION,
|
||||||
SUPPORT_WHITE_VALUE,
|
|
||||||
} from "../../../src/data/light";
|
} from "../../../src/data/light";
|
||||||
import "../../../src/dialogs/more-info/more-info-content";
|
import "../../../src/dialogs/more-info/more-info-content";
|
||||||
import { getEntity } from "../../../src/fake_data/entity";
|
import { getEntity } from "../../../src/fake_data/entity";
|
||||||
@@ -32,7 +22,8 @@ const ENTITIES = [
|
|||||||
getEntity("light", "kitchen_light", "on", {
|
getEntity("light", "kitchen_light", "on", {
|
||||||
friendly_name: "Brightness Light",
|
friendly_name: "Brightness Light",
|
||||||
brightness: 200,
|
brightness: 200,
|
||||||
supported_features: SUPPORT_BRIGHTNESS,
|
supported_color_modes: [LightColorModes.BRIGHTNESS],
|
||||||
|
color_mode: LightColorModes.BRIGHTNESS,
|
||||||
}),
|
}),
|
||||||
getEntity("light", "color_temperature_light", "on", {
|
getEntity("light", "color_temperature_light", "on", {
|
||||||
friendly_name: "White Color Temperature Light",
|
friendly_name: "White Color Temperature Light",
|
||||||
@@ -40,20 +31,96 @@ const ENTITIES = [
|
|||||||
color_temp: 75,
|
color_temp: 75,
|
||||||
min_mireds: 30,
|
min_mireds: 30,
|
||||||
max_mireds: 150,
|
max_mireds: 150,
|
||||||
supported_features: SUPPORT_BRIGHTNESS + SUPPORT_COLOR_TEMP,
|
supported_color_modes: [
|
||||||
|
LightColorModes.BRIGHTNESS,
|
||||||
|
LightColorModes.COLOR_TEMP,
|
||||||
|
],
|
||||||
|
color_mode: LightColorModes.COLOR_TEMP,
|
||||||
}),
|
}),
|
||||||
getEntity("light", "color_effectslight", "on", {
|
getEntity("light", "color_hs_light", "on", {
|
||||||
friendly_name: "Color Effets Light",
|
friendly_name: "Color HS Light",
|
||||||
brightness: 255,
|
brightness: 255,
|
||||||
hs_color: [30, 100],
|
hs_color: [30, 100],
|
||||||
white_value: 36,
|
rgb_color: [30, 100, 255],
|
||||||
supported_features:
|
min_mireds: 30,
|
||||||
SUPPORT_BRIGHTNESS +
|
max_mireds: 150,
|
||||||
SUPPORT_EFFECT +
|
supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
|
||||||
SUPPORT_FLASH +
|
supported_color_modes: [
|
||||||
SUPPORT_COLOR +
|
LightColorModes.BRIGHTNESS,
|
||||||
SUPPORT_TRANSITION +
|
LightColorModes.COLOR_TEMP,
|
||||||
SUPPORT_WHITE_VALUE,
|
LightColorModes.HS,
|
||||||
|
],
|
||||||
|
color_mode: LightColorModes.HS,
|
||||||
|
effect_list: ["random", "colorloop"],
|
||||||
|
}),
|
||||||
|
getEntity("light", "color_rgb_ct_light", "on", {
|
||||||
|
friendly_name: "Color RGB + CT Light",
|
||||||
|
brightness: 255,
|
||||||
|
color_temp: 75,
|
||||||
|
min_mireds: 30,
|
||||||
|
max_mireds: 150,
|
||||||
|
supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
|
||||||
|
supported_color_modes: [
|
||||||
|
LightColorModes.BRIGHTNESS,
|
||||||
|
LightColorModes.COLOR_TEMP,
|
||||||
|
LightColorModes.RGB,
|
||||||
|
],
|
||||||
|
color_mode: LightColorModes.COLOR_TEMP,
|
||||||
|
effect_list: ["random", "colorloop"],
|
||||||
|
}),
|
||||||
|
getEntity("light", "color_RGB_light", "on", {
|
||||||
|
friendly_name: "Color Effets Light",
|
||||||
|
brightness: 255,
|
||||||
|
rgb_color: [30, 100, 255],
|
||||||
|
supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
|
||||||
|
supported_color_modes: [LightColorModes.BRIGHTNESS, LightColorModes.RGB],
|
||||||
|
color_mode: LightColorModes.RGB,
|
||||||
|
effect_list: ["random", "colorloop"],
|
||||||
|
}),
|
||||||
|
getEntity("light", "color_rgbw_light", "on", {
|
||||||
|
friendly_name: "Color RGBW Light",
|
||||||
|
brightness: 255,
|
||||||
|
rgbw_color: [30, 100, 255, 125],
|
||||||
|
min_mireds: 30,
|
||||||
|
max_mireds: 150,
|
||||||
|
supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
|
||||||
|
supported_color_modes: [
|
||||||
|
LightColorModes.BRIGHTNESS,
|
||||||
|
LightColorModes.COLOR_TEMP,
|
||||||
|
LightColorModes.RGBW,
|
||||||
|
],
|
||||||
|
color_mode: LightColorModes.RGBW,
|
||||||
|
effect_list: ["random", "colorloop"],
|
||||||
|
}),
|
||||||
|
getEntity("light", "color_rgbww_light", "on", {
|
||||||
|
friendly_name: "Color RGBWW Light",
|
||||||
|
brightness: 255,
|
||||||
|
rgbww_color: [30, 100, 255, 125, 10],
|
||||||
|
min_mireds: 30,
|
||||||
|
max_mireds: 150,
|
||||||
|
supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
|
||||||
|
supported_color_modes: [
|
||||||
|
LightColorModes.BRIGHTNESS,
|
||||||
|
LightColorModes.COLOR_TEMP,
|
||||||
|
LightColorModes.RGBWW,
|
||||||
|
],
|
||||||
|
color_mode: LightColorModes.RGBWW,
|
||||||
|
effect_list: ["random", "colorloop"],
|
||||||
|
}),
|
||||||
|
getEntity("light", "color_xy_light", "on", {
|
||||||
|
friendly_name: "Color XY Light",
|
||||||
|
brightness: 255,
|
||||||
|
xy_color: [30, 100],
|
||||||
|
rgb_color: [30, 100, 255],
|
||||||
|
min_mireds: 30,
|
||||||
|
max_mireds: 150,
|
||||||
|
supported_features: SUPPORT_EFFECT + SUPPORT_FLASH + SUPPORT_TRANSITION,
|
||||||
|
supported_color_modes: [
|
||||||
|
LightColorModes.BRIGHTNESS,
|
||||||
|
LightColorModes.COLOR_TEMP,
|
||||||
|
LightColorModes.XY,
|
||||||
|
],
|
||||||
|
color_mode: LightColorModes.XY,
|
||||||
effect_list: ["random", "colorloop"],
|
effect_list: ["random", "colorloop"],
|
||||||
}),
|
}),
|
||||||
];
|
];
|
||||||
@@ -81,4 +148,8 @@ class DemoMoreInfoLight extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
customElements.define("demo-more-info-light", DemoMoreInfoLight);
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-more-info-light": DemoMoreInfoLight;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
import "@material/mwc-button";
|
import "@material/mwc-button";
|
||||||
import { customElement, html, LitElement, TemplateResult } from "lit-element";
|
import { html, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement } from "lit/decorators";
|
||||||
import "../../../src/components/ha-card";
|
import "../../../src/components/ha-card";
|
||||||
import { ActionHandlerEvent } from "../../../src/data/lovelace";
|
import { ActionHandlerEvent } from "../../../src/data/lovelace";
|
||||||
import { actionHandler } from "../../../src/panels/lovelace/common/directives/action-handler-directive";
|
import { actionHandler } from "../../../src/panels/lovelace/common/directives/action-handler-directive";
|
||||||
|
@@ -111,29 +111,9 @@ class HaGallery extends PolymerElement {
|
|||||||
</template>
|
</template>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
|
|
||||||
<ha-card header="More Info Demos">
|
<ha-card header="Other Demos">
|
||||||
<div class="card-content intro">
|
<div class="card-content intro"></div>
|
||||||
<p>
|
<template is="dom-repeat" items="[[_restDemos]]">
|
||||||
More info screens show up when an entity is clicked.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<template is="dom-repeat" items="[[_moreInfoDemos]]">
|
|
||||||
<a href="#[[item]]">
|
|
||||||
<paper-item>
|
|
||||||
<paper-item-body>{{ item }}</paper-item-body>
|
|
||||||
<ha-icon icon="hass:chevron-right"></ha-icon>
|
|
||||||
</paper-item>
|
|
||||||
</a>
|
|
||||||
</template>
|
|
||||||
</ha-card>
|
|
||||||
|
|
||||||
<ha-card header="Util Demos">
|
|
||||||
<div class="card-content intro">
|
|
||||||
<p>
|
|
||||||
Test pages for our utility functions.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
<template is="dom-repeat" items="[[_utilDemos]]">
|
|
||||||
<a href="#[[item]]">
|
<a href="#[[item]]">
|
||||||
<paper-item>
|
<paper-item>
|
||||||
<paper-item-body>{{ item }}</paper-item-body>
|
<paper-item-body>{{ item }}</paper-item-body>
|
||||||
@@ -178,13 +158,9 @@ class HaGallery extends PolymerElement {
|
|||||||
type: Array,
|
type: Array,
|
||||||
computed: "_computeLovelace(_demos)",
|
computed: "_computeLovelace(_demos)",
|
||||||
},
|
},
|
||||||
_moreInfoDemos: {
|
_restDemos: {
|
||||||
type: Array,
|
type: Array,
|
||||||
computed: "_computeMoreInfos(_demos)",
|
computed: "_computeRest(_demos)",
|
||||||
},
|
|
||||||
_utilDemos: {
|
|
||||||
type: Array,
|
|
||||||
computed: "_computeUtil(_demos)",
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@@ -237,12 +213,8 @@ class HaGallery extends PolymerElement {
|
|||||||
return demos.filter((demo) => demo.includes("hui"));
|
return demos.filter((demo) => demo.includes("hui"));
|
||||||
}
|
}
|
||||||
|
|
||||||
_computeMoreInfos(demos) {
|
_computeRest(demos) {
|
||||||
return demos.filter((demo) => demo.includes("more-info"));
|
return demos.filter((demo) => !demo.includes("hui"));
|
||||||
}
|
|
||||||
|
|
||||||
_computeUtil(demos) {
|
|
||||||
return demos.filter((demo) => demo.includes("util"));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,12 +1,6 @@
|
|||||||
import { mdiArrowUpBoldCircle, mdiPuzzle } from "@mdi/js";
|
import { mdiArrowUpBoldCircle, mdiPuzzle } from "@mdi/js";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { property } from "lit/decorators";
|
||||||
CSSResultArray,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { atLeastVersion } from "../../../src/common/config/version";
|
import { atLeastVersion } from "../../../src/common/config/version";
|
||||||
import { navigate } from "../../../src/common/navigate";
|
import { navigate } from "../../../src/common/navigate";
|
||||||
@@ -15,6 +9,7 @@ import {
|
|||||||
HassioAddonInfo,
|
HassioAddonInfo,
|
||||||
HassioAddonRepository,
|
HassioAddonRepository,
|
||||||
} from "../../../src/data/hassio/addon";
|
} from "../../../src/data/hassio/addon";
|
||||||
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import { HomeAssistant } from "../../../src/types";
|
import { HomeAssistant } from "../../../src/types";
|
||||||
import "../components/hassio-card-content";
|
import "../components/hassio-card-content";
|
||||||
import { filterAndSort } from "../components/hassio-filter-addons";
|
import { filterAndSort } from "../components/hassio-filter-addons";
|
||||||
@@ -23,6 +18,8 @@ import { hassioStyle } from "../resources/hassio-style";
|
|||||||
class HassioAddonRepositoryEl extends LitElement {
|
class HassioAddonRepositoryEl extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public repo!: HassioAddonRepository;
|
@property({ attribute: false }) public repo!: HassioAddonRepository;
|
||||||
|
|
||||||
@property({ attribute: false }) public addons!: HassioAddonInfo[];
|
@property({ attribute: false }) public addons!: HassioAddonInfo[];
|
||||||
@@ -44,9 +41,7 @@ class HassioAddonRepositoryEl extends LitElement {
|
|||||||
const repo = this.repo;
|
const repo = this.repo;
|
||||||
let _addons = this.addons;
|
let _addons = this.addons;
|
||||||
if (!this.hass.userData?.showAdvanced) {
|
if (!this.hass.userData?.showAdvanced) {
|
||||||
_addons = _addons.filter((addon) => {
|
_addons = _addons.filter((addon) => !addon.advanced);
|
||||||
return !addon.advanced;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
const addons = this._getAddons(_addons, this.filter);
|
const addons = this._getAddons(_addons, this.filter);
|
||||||
|
|
||||||
@@ -54,16 +49,18 @@ class HassioAddonRepositoryEl extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<p class="description">
|
<p class="description">
|
||||||
No results found in "${repo.name}."
|
${this.supervisor.localize(
|
||||||
|
"store.no_results_found",
|
||||||
|
"repository",
|
||||||
|
repo.name
|
||||||
|
)}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<h1>
|
<h1>${repo.name}</h1>
|
||||||
${repo.name}
|
|
||||||
</h1>
|
|
||||||
<div class="card-group">
|
<div class="card-group">
|
||||||
${addons.map(
|
${addons.map(
|
||||||
(addon) => html`
|
(addon) => html`
|
||||||
@@ -83,11 +80,13 @@ class HassioAddonRepositoryEl extends LitElement {
|
|||||||
: mdiPuzzle}
|
: mdiPuzzle}
|
||||||
.iconTitle=${addon.installed
|
.iconTitle=${addon.installed
|
||||||
? addon.update_available
|
? addon.update_available
|
||||||
? "New version available"
|
? this.supervisor.localize(
|
||||||
: "Add-on is installed"
|
"common.new_version_available"
|
||||||
|
)
|
||||||
|
: this.supervisor.localize("addon.installed")
|
||||||
: addon.available
|
: addon.available
|
||||||
? "Add-on is not installed"
|
? this.supervisor.localize("addon.not_installed")
|
||||||
: "Add-on is not available on your system"}
|
: this.supervisor.localize("addon.not_available")}
|
||||||
.iconClass=${addon.installed
|
.iconClass=${addon.installed
|
||||||
? addon.update_available
|
? addon.update_available
|
||||||
? "update"
|
? "update"
|
||||||
@@ -121,10 +120,10 @@ class HassioAddonRepositoryEl extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private _addonTapped(ev) {
|
private _addonTapped(ev) {
|
||||||
navigate(this, `/hassio/addon/${ev.currentTarget.addon.slug}`);
|
navigate(`/hassio/addon/${ev.currentTarget.addon.slug}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResultArray {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
css`
|
css`
|
||||||
|
@@ -4,17 +4,19 @@ import "@material/mwc-list/mwc-list-item";
|
|||||||
import { mdiDotsVertical } from "@mdi/js";
|
import { mdiDotsVertical } from "@mdi/js";
|
||||||
import {
|
import {
|
||||||
css,
|
css,
|
||||||
CSSResult,
|
CSSResultGroup,
|
||||||
internalProperty,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
property,
|
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
} from "lit-element";
|
TemplateResult,
|
||||||
import { html, TemplateResult } from "lit-html";
|
} from "lit";
|
||||||
|
import { property, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { atLeastVersion } from "../../../src/common/config/version";
|
import { atLeastVersion } from "../../../src/common/config/version";
|
||||||
import { fireEvent } from "../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../src/common/dom/fire_event";
|
||||||
|
import { navigate } from "../../../src/common/navigate";
|
||||||
import "../../../src/common/search/search-input";
|
import "../../../src/common/search/search-input";
|
||||||
|
import { extractSearchParam } from "../../../src/common/url/search-params";
|
||||||
import "../../../src/components/ha-button-menu";
|
import "../../../src/components/ha-button-menu";
|
||||||
import "../../../src/components/ha-svg-icon";
|
import "../../../src/components/ha-svg-icon";
|
||||||
import {
|
import {
|
||||||
@@ -56,7 +58,7 @@ class HassioAddonStore extends LitElement {
|
|||||||
|
|
||||||
@property({ attribute: false }) public route!: Route;
|
@property({ attribute: false }) public route!: Route;
|
||||||
|
|
||||||
@internalProperty() private _filter?: string;
|
@state() private _filter?: string;
|
||||||
|
|
||||||
public async refreshData() {
|
public async refreshData() {
|
||||||
await reloadHassioAddons(this.hass);
|
await reloadHassioAddons(this.hass);
|
||||||
@@ -77,13 +79,14 @@ class HassioAddonStore extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
<hass-tabs-subpage
|
<hass-tabs-subpage
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.localizeFunc=${this.supervisor.localize}
|
||||||
.narrow=${this.narrow}
|
.narrow=${this.narrow}
|
||||||
.route=${this.route}
|
.route=${this.route}
|
||||||
hassio
|
|
||||||
main-page
|
|
||||||
.tabs=${supervisorTabs}
|
.tabs=${supervisorTabs}
|
||||||
|
main-page
|
||||||
|
supervisor
|
||||||
>
|
>
|
||||||
<span slot="header">Add-on Store</span>
|
<span slot="header"> ${this.supervisor.localize("panel.store")} </span>
|
||||||
<ha-button-menu
|
<ha-button-menu
|
||||||
corner="BOTTOM_START"
|
corner="BOTTOM_START"
|
||||||
slot="toolbar-icon"
|
slot="toolbar-icon"
|
||||||
@@ -93,15 +96,15 @@ class HassioAddonStore extends LitElement {
|
|||||||
<ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon>
|
||||||
</mwc-icon-button>
|
</mwc-icon-button>
|
||||||
<mwc-list-item>
|
<mwc-list-item>
|
||||||
Repositories
|
${this.supervisor.localize("store.repositories")}
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
<mwc-list-item>
|
<mwc-list-item>
|
||||||
Reload
|
${this.supervisor.localize("common.reload")}
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
${this.hass.userData?.showAdvanced &&
|
${this.hass.userData?.showAdvanced &&
|
||||||
atLeastVersion(this.hass.config.version, 0, 117)
|
atLeastVersion(this.hass.config.version, 0, 117)
|
||||||
? html`<mwc-list-item>
|
? html`<mwc-list-item>
|
||||||
Registries
|
${this.supervisor.localize("store.registries")}
|
||||||
</mwc-list-item>`
|
</mwc-list-item>`
|
||||||
: ""}
|
: ""}
|
||||||
</ha-button-menu>
|
</ha-button-menu>
|
||||||
@@ -122,11 +125,9 @@ class HassioAddonStore extends LitElement {
|
|||||||
${!this.hass.userData?.showAdvanced
|
${!this.hass.userData?.showAdvanced
|
||||||
? html`
|
? html`
|
||||||
<div class="advanced">
|
<div class="advanced">
|
||||||
Missing add-ons? Enable advanced mode on
|
|
||||||
<a href="/profile" target="_top">
|
<a href="/profile" target="_top">
|
||||||
your profile page
|
${this.supervisor.localize("store.missing_addons")}
|
||||||
</a>
|
</a>
|
||||||
.
|
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
@@ -136,6 +137,12 @@ class HassioAddonStore extends LitElement {
|
|||||||
|
|
||||||
protected firstUpdated(changedProps: PropertyValues) {
|
protected firstUpdated(changedProps: PropertyValues) {
|
||||||
super.firstUpdated(changedProps);
|
super.firstUpdated(changedProps);
|
||||||
|
const repositoryUrl = extractSearchParam("repository_url");
|
||||||
|
navigate("/hassio/store", { replace: true });
|
||||||
|
if (repositoryUrl) {
|
||||||
|
this._manageRepositories(repositoryUrl);
|
||||||
|
}
|
||||||
|
|
||||||
this.addEventListener("hass-api-called", (ev) => this.apiCalled(ev));
|
this.addEventListener("hass-api-called", (ev) => this.apiCalled(ev));
|
||||||
this._loadData();
|
this._loadData();
|
||||||
}
|
}
|
||||||
@@ -145,8 +152,8 @@ class HassioAddonStore extends LitElement {
|
|||||||
repositories: HassioAddonRepository[],
|
repositories: HassioAddonRepository[],
|
||||||
addons: HassioAddonInfo[],
|
addons: HassioAddonInfo[],
|
||||||
filter?: string
|
filter?: string
|
||||||
) => {
|
) =>
|
||||||
return repositories.sort(sortRepos).map((repo) => {
|
repositories.sort(sortRepos).map((repo) => {
|
||||||
const filteredAddons = addons.filter(
|
const filteredAddons = addons.filter(
|
||||||
(addon) => addon.repository === repo.slug
|
(addon) => addon.repository === repo.slug
|
||||||
);
|
);
|
||||||
@@ -158,17 +165,17 @@ class HassioAddonStore extends LitElement {
|
|||||||
.repo=${repo}
|
.repo=${repo}
|
||||||
.addons=${filteredAddons}
|
.addons=${filteredAddons}
|
||||||
.filter=${filter!}
|
.filter=${filter!}
|
||||||
|
.supervisor=${this.supervisor}
|
||||||
></hassio-addon-repository>
|
></hassio-addon-repository>
|
||||||
`
|
`
|
||||||
: html``;
|
: html``;
|
||||||
});
|
})
|
||||||
}
|
|
||||||
);
|
);
|
||||||
|
|
||||||
private _handleAction(ev: CustomEvent<ActionDetail>) {
|
private _handleAction(ev: CustomEvent<ActionDetail>) {
|
||||||
switch (ev.detail.index) {
|
switch (ev.detail.index) {
|
||||||
case 0:
|
case 0:
|
||||||
this._manageRepositories();
|
this._manageRepositoriesClicked();
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
this.refreshData();
|
this.refreshData();
|
||||||
@@ -185,21 +192,25 @@ class HassioAddonStore extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _manageRepositories() {
|
private _manageRepositoriesClicked() {
|
||||||
|
this._manageRepositories();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _manageRepositories(url?: string) {
|
||||||
showRepositoriesDialog(this, {
|
showRepositoriesDialog(this, {
|
||||||
repos: this.supervisor.addon.repositories,
|
supervisor: this.supervisor,
|
||||||
loadData: () => this._loadData(),
|
url,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _manageRegistries() {
|
private async _manageRegistries() {
|
||||||
showRegistriesDialog(this);
|
showRegistriesDialog(this, { supervisor: this.supervisor });
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _loadData() {
|
private async _loadData() {
|
||||||
fireEvent(this, "supervisor-colllection-refresh", { colllection: "addon" });
|
fireEvent(this, "supervisor-collection-refresh", { collection: "addon" });
|
||||||
fireEvent(this, "supervisor-colllection-refresh", {
|
fireEvent(this, "supervisor-collection-refresh", {
|
||||||
colllection: "supervisor",
|
collection: "supervisor",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -207,7 +218,7 @@ class HassioAddonStore extends LitElement {
|
|||||||
this._filter = e.detail.value;
|
this._filter = e.detail.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
hassio-addon-repository {
|
hassio-addon-repository {
|
||||||
margin-top: 24px;
|
margin-top: 24px;
|
||||||
|
@@ -4,15 +4,13 @@ import "@polymer/paper-item/paper-item";
|
|||||||
import "@polymer/paper-listbox/paper-listbox";
|
import "@polymer/paper-listbox/paper-listbox";
|
||||||
import {
|
import {
|
||||||
css,
|
css,
|
||||||
CSSResult,
|
CSSResultGroup,
|
||||||
customElement,
|
|
||||||
html,
|
html,
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
LitElement,
|
||||||
property,
|
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit-element";
|
} from "lit";
|
||||||
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import "web-animations-js/web-animations-next-lite.min";
|
import "web-animations-js/web-animations-next-lite.min";
|
||||||
import "../../../../src/components/buttons/ha-progress-button";
|
import "../../../../src/components/buttons/ha-progress-button";
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
@@ -25,6 +23,7 @@ import {
|
|||||||
fetchHassioHardwareAudio,
|
fetchHassioHardwareAudio,
|
||||||
HassioHardwareAudioDevice,
|
HassioHardwareAudioDevice,
|
||||||
} from "../../../../src/data/hassio/hardware";
|
} from "../../../../src/data/hassio/hardware";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import { haStyle } from "../../../../src/resources/styles";
|
import { haStyle } from "../../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart";
|
import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart";
|
||||||
@@ -34,26 +33,32 @@ import { hassioStyle } from "../../resources/hassio-style";
|
|||||||
class HassioAddonAudio extends LitElement {
|
class HassioAddonAudio extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public addon!: HassioAddonDetails;
|
@property({ attribute: false }) public addon!: HassioAddonDetails;
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
@state() private _error?: string;
|
||||||
|
|
||||||
@internalProperty() private _inputDevices?: HassioHardwareAudioDevice[];
|
@state() private _inputDevices?: HassioHardwareAudioDevice[];
|
||||||
|
|
||||||
@internalProperty() private _outputDevices?: HassioHardwareAudioDevice[];
|
@state() private _outputDevices?: HassioHardwareAudioDevice[];
|
||||||
|
|
||||||
@internalProperty() private _selectedInput!: null | string;
|
@state() private _selectedInput!: null | string;
|
||||||
|
|
||||||
@internalProperty() private _selectedOutput!: null | string;
|
@state() private _selectedOutput!: null | string;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
<ha-card header="Audio">
|
<ha-card
|
||||||
|
.header=${this.supervisor.localize("addon.configuration.audio.header")}
|
||||||
|
>
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
|
${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
|
||||||
|
|
||||||
<paper-dropdown-menu
|
<paper-dropdown-menu
|
||||||
label="Input"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.configuration.audio.input"
|
||||||
|
)}
|
||||||
@iron-select=${this._setInputDevice}
|
@iron-select=${this._setInputDevice}
|
||||||
>
|
>
|
||||||
<paper-listbox
|
<paper-listbox
|
||||||
@@ -62,17 +67,19 @@ class HassioAddonAudio extends LitElement {
|
|||||||
.selected=${this._selectedInput!}
|
.selected=${this._selectedInput!}
|
||||||
>
|
>
|
||||||
${this._inputDevices &&
|
${this._inputDevices &&
|
||||||
this._inputDevices.map((item) => {
|
this._inputDevices.map(
|
||||||
return html`
|
(item) => html`
|
||||||
<paper-item device=${item.device || ""}
|
<paper-item device=${item.device || ""}>
|
||||||
>${item.name}</paper-item
|
${item.name}
|
||||||
>
|
</paper-item>
|
||||||
`;
|
`
|
||||||
})}
|
)}
|
||||||
</paper-listbox>
|
</paper-listbox>
|
||||||
</paper-dropdown-menu>
|
</paper-dropdown-menu>
|
||||||
<paper-dropdown-menu
|
<paper-dropdown-menu
|
||||||
label="Output"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.configuration.audio.output"
|
||||||
|
)}
|
||||||
@iron-select=${this._setOutputDevice}
|
@iron-select=${this._setOutputDevice}
|
||||||
>
|
>
|
||||||
<paper-listbox
|
<paper-listbox
|
||||||
@@ -81,26 +88,26 @@ class HassioAddonAudio extends LitElement {
|
|||||||
.selected=${this._selectedOutput!}
|
.selected=${this._selectedOutput!}
|
||||||
>
|
>
|
||||||
${this._outputDevices &&
|
${this._outputDevices &&
|
||||||
this._outputDevices.map((item) => {
|
this._outputDevices.map(
|
||||||
return html`
|
(item) => html`
|
||||||
<paper-item device=${item.device || ""}
|
<paper-item device=${item.device || ""}
|
||||||
>${item.name}</paper-item
|
>${item.name}</paper-item
|
||||||
>
|
>
|
||||||
`;
|
`
|
||||||
})}
|
)}
|
||||||
</paper-listbox>
|
</paper-listbox>
|
||||||
</paper-dropdown-menu>
|
</paper-dropdown-menu>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
<ha-progress-button @click=${this._saveSettings}>
|
<ha-progress-button @click=${this._saveSettings}>
|
||||||
Save
|
${this.supervisor.localize("common.save")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
@@ -152,7 +159,7 @@ class HassioAddonAudio extends LitElement {
|
|||||||
|
|
||||||
const noDevice: HassioHardwareAudioDevice = {
|
const noDevice: HassioHardwareAudioDevice = {
|
||||||
device: "default",
|
device: "default",
|
||||||
name: "Default",
|
name: this.supervisor.localize("addon.configuration.audio.default"),
|
||||||
};
|
};
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -189,7 +196,7 @@ class HassioAddonAudio extends LitElement {
|
|||||||
try {
|
try {
|
||||||
await setHassioAddonOption(this.hass, this.addon.slug, data);
|
await setHassioAddonOption(this.hass, this.addon.slug, data);
|
||||||
if (this.addon?.state === "started") {
|
if (this.addon?.state === "started") {
|
||||||
await suggestAddonRestart(this, this.hass, this.addon);
|
await suggestAddonRestart(this, this.hass, this.supervisor, this.addon);
|
||||||
}
|
}
|
||||||
} catch {
|
} catch {
|
||||||
this._error = "Failed to set addon audio device";
|
this._error = "Failed to set addon audio device";
|
||||||
|
@@ -1,14 +1,8 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../../src/components/ha-circular-progress";
|
import "../../../../src/components/ha-circular-progress";
|
||||||
import { HassioAddonDetails } from "../../../../src/data/hassio/addon";
|
import { HassioAddonDetails } from "../../../../src/data/hassio/addon";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import { haStyle } from "../../../../src/resources/styles";
|
import { haStyle } from "../../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
import { hassioStyle } from "../../resources/hassio-style";
|
import { hassioStyle } from "../../resources/hassio-style";
|
||||||
@@ -20,6 +14,8 @@ import "./hassio-addon-network";
|
|||||||
class HassioAddonConfigDashboard extends LitElement {
|
class HassioAddonConfigDashboard extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public addon?: HassioAddonDetails;
|
@property({ attribute: false }) public addon?: HassioAddonDetails;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
@@ -39,6 +35,7 @@ class HassioAddonConfigDashboard extends LitElement {
|
|||||||
<hassio-addon-config
|
<hassio-addon-config
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.addon=${this.addon}
|
.addon=${this.addon}
|
||||||
|
.supervisor=${this.supervisor}
|
||||||
></hassio-addon-config>
|
></hassio-addon-config>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
@@ -47,6 +44,7 @@ class HassioAddonConfigDashboard extends LitElement {
|
|||||||
<hassio-addon-network
|
<hassio-addon-network
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.addon=${this.addon}
|
.addon=${this.addon}
|
||||||
|
.supervisor=${this.supervisor}
|
||||||
></hassio-addon-network>
|
></hassio-addon-network>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
@@ -55,16 +53,17 @@ class HassioAddonConfigDashboard extends LitElement {
|
|||||||
<hassio-addon-audio
|
<hassio-addon-audio
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.addon=${this.addon}
|
.addon=${this.addon}
|
||||||
|
.supervisor=${this.supervisor}
|
||||||
></hassio-addon-audio>
|
></hassio-addon-audio>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
`
|
`
|
||||||
: "This add-on does not expose configuration for you to mess with.... 👋"}
|
: this.supervisor.localize("addon.configuration.no_configuration")}
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
|
@@ -3,18 +3,16 @@ import { ActionDetail } from "@material/mwc-list";
|
|||||||
import "@material/mwc-list/mwc-list-item";
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import { mdiDotsVertical } from "@mdi/js";
|
import { mdiDotsVertical } from "@mdi/js";
|
||||||
import "@polymer/iron-autogrow-textarea/iron-autogrow-textarea";
|
import "@polymer/iron-autogrow-textarea/iron-autogrow-textarea";
|
||||||
|
import { DEFAULT_SCHEMA, Type } from "js-yaml";
|
||||||
import {
|
import {
|
||||||
css,
|
css,
|
||||||
CSSResult,
|
CSSResultGroup,
|
||||||
customElement,
|
|
||||||
html,
|
html,
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
LitElement,
|
||||||
property,
|
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
query,
|
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit-element";
|
} from "lit";
|
||||||
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import "../../../../src/components/buttons/ha-progress-button";
|
import "../../../../src/components/buttons/ha-progress-button";
|
||||||
@@ -30,8 +28,10 @@ import {
|
|||||||
HassioAddonDetails,
|
HassioAddonDetails,
|
||||||
HassioAddonSetOptionParams,
|
HassioAddonSetOptionParams,
|
||||||
setHassioAddonOption,
|
setHassioAddonOption,
|
||||||
|
validateHassioAddonOption,
|
||||||
} from "../../../../src/data/hassio/addon";
|
} from "../../../../src/data/hassio/addon";
|
||||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import { showConfirmationDialog } from "../../../../src/dialogs/generic/show-dialog-box";
|
import { showConfirmationDialog } from "../../../../src/dialogs/generic/show-dialog-box";
|
||||||
import { haStyle } from "../../../../src/resources/styles";
|
import { haStyle } from "../../../../src/resources/styles";
|
||||||
import type { HomeAssistant } from "../../../../src/types";
|
import type { HomeAssistant } from "../../../../src/types";
|
||||||
@@ -40,32 +40,46 @@ import { hassioStyle } from "../../resources/hassio-style";
|
|||||||
|
|
||||||
const SUPPORTED_UI_TYPES = ["string", "select", "boolean", "integer", "float"];
|
const SUPPORTED_UI_TYPES = ["string", "select", "boolean", "integer", "float"];
|
||||||
|
|
||||||
|
const ADDON_YAML_SCHEMA = DEFAULT_SCHEMA.extend([
|
||||||
|
new Type("!secret", {
|
||||||
|
kind: "scalar",
|
||||||
|
construct: (data) => `!secret ${data}`,
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
@customElement("hassio-addon-config")
|
@customElement("hassio-addon-config")
|
||||||
class HassioAddonConfig extends LitElement {
|
class HassioAddonConfig extends LitElement {
|
||||||
@property({ attribute: false }) public addon!: HassioAddonDetails;
|
@property({ attribute: false }) public addon!: HassioAddonDetails;
|
||||||
|
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ type: Boolean }) private _configHasChanged = false;
|
@property({ type: Boolean }) private _configHasChanged = false;
|
||||||
|
|
||||||
@property({ type: Boolean }) private _valid = true;
|
@property({ type: Boolean }) private _valid = true;
|
||||||
|
|
||||||
@internalProperty() private _canShowSchema = false;
|
@state() private _canShowSchema = false;
|
||||||
|
|
||||||
@internalProperty() private _showOptional = false;
|
@state() private _showOptional = false;
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
@state() private _error?: string;
|
||||||
|
|
||||||
@internalProperty() private _options?: Record<string, unknown>;
|
@state() private _options?: Record<string, unknown>;
|
||||||
|
|
||||||
@internalProperty() private _yamlMode = false;
|
@state() private _yamlMode = false;
|
||||||
|
|
||||||
@query("ha-yaml-editor") private _editor?: HaYamlEditor;
|
@query("ha-yaml-editor") private _editor?: HaYamlEditor;
|
||||||
|
|
||||||
|
public computeLabel = (entry: HaFormSchema): string =>
|
||||||
|
this.addon.translations[this.hass.language]?.configuration?.[entry.name]
|
||||||
|
?.name ||
|
||||||
|
this.addon.translations.en?.configuration?.[entry.name].name ||
|
||||||
|
entry.name;
|
||||||
|
|
||||||
private _filteredShchema = memoizeOne(
|
private _filteredShchema = memoizeOne(
|
||||||
(options: Record<string, unknown>, schema: HaFormSchema[]) => {
|
(options: Record<string, unknown>, schema: HaFormSchema[]) =>
|
||||||
return schema.filter((entry) => entry.name in options || entry.required);
|
schema.filter((entry) => entry.name in options || entry.required)
|
||||||
}
|
|
||||||
);
|
);
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
@@ -81,17 +95,25 @@ class HassioAddonConfig extends LitElement {
|
|||||||
<h1>${this.addon.name}</h1>
|
<h1>${this.addon.name}</h1>
|
||||||
<ha-card>
|
<ha-card>
|
||||||
<div class="header">
|
<div class="header">
|
||||||
<h2>Configuration</h2>
|
<h2>
|
||||||
|
${this.supervisor.localize("addon.configuration.options.header")}
|
||||||
|
</h2>
|
||||||
<div class="card-menu">
|
<div class="card-menu">
|
||||||
<ha-button-menu corner="BOTTOM_START" @action=${this._handleAction}>
|
<ha-button-menu corner="BOTTOM_START" @action=${this._handleAction}>
|
||||||
<mwc-icon-button slot="trigger">
|
<mwc-icon-button slot="trigger">
|
||||||
<ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiDotsVertical}></ha-svg-icon>
|
||||||
</mwc-icon-button>
|
</mwc-icon-button>
|
||||||
<mwc-list-item .disabled=${!this._canShowSchema}>
|
<mwc-list-item .disabled=${!this._canShowSchema}>
|
||||||
${this._yamlMode ? "Edit in UI" : "Edit in YAML"}
|
${this._yamlMode
|
||||||
|
? this.supervisor.localize(
|
||||||
|
"addon.configuration.options.edit_in_ui"
|
||||||
|
)
|
||||||
|
: this.supervisor.localize(
|
||||||
|
"addon.configuration.options.edit_in_yaml"
|
||||||
|
)}
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
<mwc-list-item class="warning">
|
<mwc-list-item class="warning">
|
||||||
Reset to defaults
|
${this.supervisor.localize("common.reset_defaults")}
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
</ha-button-menu>
|
</ha-button-menu>
|
||||||
</div>
|
</div>
|
||||||
@@ -102,6 +124,7 @@ class HassioAddonConfig extends LitElement {
|
|||||||
? html`<ha-form
|
? html`<ha-form
|
||||||
.data=${this._options!}
|
.data=${this._options!}
|
||||||
@value-changed=${this._configChanged}
|
@value-changed=${this._configChanged}
|
||||||
|
.computeLabel=${this.computeLabel}
|
||||||
.schema=${this._showOptional
|
.schema=${this._showOptional
|
||||||
? this.addon.schema!
|
? this.addon.schema!
|
||||||
: this._filteredShchema(
|
: this._filteredShchema(
|
||||||
@@ -111,18 +134,27 @@ class HassioAddonConfig extends LitElement {
|
|||||||
></ha-form>`
|
></ha-form>`
|
||||||
: html` <ha-yaml-editor
|
: html` <ha-yaml-editor
|
||||||
@value-changed=${this._configChanged}
|
@value-changed=${this._configChanged}
|
||||||
|
.schema=${ADDON_YAML_SCHEMA}
|
||||||
></ha-yaml-editor>`}
|
></ha-yaml-editor>`}
|
||||||
${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
|
${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
|
||||||
${!this._yamlMode ||
|
${!this._yamlMode ||
|
||||||
(this._canShowSchema && this.addon.schema) ||
|
(this._canShowSchema && this.addon.schema) ||
|
||||||
this._valid
|
this._valid
|
||||||
? ""
|
? ""
|
||||||
: html` <div class="errors">Invalid YAML</div> `}
|
: html`
|
||||||
|
<div class="errors">
|
||||||
|
${this.supervisor.localize(
|
||||||
|
"addon.configuration.options.invalid_yaml"
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
`}
|
||||||
</div>
|
</div>
|
||||||
${hasHiddenOptions
|
${hasHiddenOptions
|
||||||
? html`<ha-formfield
|
? html`<ha-formfield
|
||||||
class="show-additional"
|
class="show-additional"
|
||||||
label="Show unused optional configuration options"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.configuration.options.show_unused_optional"
|
||||||
|
)}
|
||||||
>
|
>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._toggleOptional}
|
@change=${this._toggleOptional}
|
||||||
@@ -136,7 +168,7 @@ class HassioAddonConfig extends LitElement {
|
|||||||
@click=${this._saveTapped}
|
@click=${this._saveTapped}
|
||||||
.disabled=${!this._configHasChanged || !this._valid}
|
.disabled=${!this._configHasChanged || !this._valid}
|
||||||
>
|
>
|
||||||
Save
|
${this.supervisor.localize("common.save")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
@@ -201,10 +233,10 @@ class HassioAddonConfig extends LitElement {
|
|||||||
button.progress = true;
|
button.progress = true;
|
||||||
|
|
||||||
const confirmed = await showConfirmationDialog(this, {
|
const confirmed = await showConfirmationDialog(this, {
|
||||||
title: this.addon.name,
|
title: this.supervisor.localize("confirm.reset_options.title"),
|
||||||
text: "Are you sure you want to reset all your options?",
|
text: this.supervisor.localize("confirm.reset_options.text"),
|
||||||
confirmText: "reset options",
|
confirmText: this.supervisor.localize("common.reset_options"),
|
||||||
dismissText: "no",
|
dismissText: this.supervisor.localize("common.cancel"),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!confirmed) {
|
if (!confirmed) {
|
||||||
@@ -226,43 +258,56 @@ class HassioAddonConfig extends LitElement {
|
|||||||
};
|
};
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to reset addon configuration, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.common.update_available",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
button.progress = false;
|
button.progress = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _saveTapped(ev: CustomEvent): Promise<void> {
|
private async _saveTapped(ev: CustomEvent): Promise<void> {
|
||||||
const button = ev.currentTarget as any;
|
const button = ev.currentTarget as any;
|
||||||
|
const eventdata = {
|
||||||
|
success: true,
|
||||||
|
response: undefined,
|
||||||
|
path: "options",
|
||||||
|
};
|
||||||
button.progress = true;
|
button.progress = true;
|
||||||
|
|
||||||
this._error = undefined;
|
this._error = undefined;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
const validation = await validateHassioAddonOption(
|
||||||
|
this.hass,
|
||||||
|
this.addon.slug,
|
||||||
|
this._editor?.value
|
||||||
|
);
|
||||||
|
if (!validation.valid) {
|
||||||
|
throw Error(validation.message);
|
||||||
|
}
|
||||||
await setHassioAddonOption(this.hass, this.addon.slug, {
|
await setHassioAddonOption(this.hass, this.addon.slug, {
|
||||||
options: this._yamlMode ? this._editor?.value : this._options,
|
options: this._yamlMode ? this._editor?.value : this._options,
|
||||||
});
|
});
|
||||||
|
|
||||||
this._configHasChanged = false;
|
this._configHasChanged = false;
|
||||||
const eventdata = {
|
|
||||||
success: true,
|
|
||||||
response: undefined,
|
|
||||||
path: "options",
|
|
||||||
};
|
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
|
||||||
if (this.addon?.state === "started") {
|
if (this.addon?.state === "started") {
|
||||||
await suggestAddonRestart(this, this.hass, this.addon);
|
await suggestAddonRestart(this, this.hass, this.supervisor, this.addon);
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to save addon configuration, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.failed_to_save",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
|
eventdata.success = false;
|
||||||
}
|
}
|
||||||
button.progress = false;
|
button.progress = false;
|
||||||
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
|
@@ -1,15 +1,13 @@
|
|||||||
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
||||||
import {
|
import {
|
||||||
css,
|
css,
|
||||||
CSSResult,
|
CSSResultGroup,
|
||||||
customElement,
|
|
||||||
html,
|
html,
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
LitElement,
|
||||||
property,
|
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit-element";
|
} from "lit";
|
||||||
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import "../../../../src/components/buttons/ha-progress-button";
|
import "../../../../src/components/buttons/ha-progress-button";
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
@@ -19,6 +17,7 @@ import {
|
|||||||
setHassioAddonOption,
|
setHassioAddonOption,
|
||||||
} from "../../../../src/data/hassio/addon";
|
} from "../../../../src/data/hassio/addon";
|
||||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import { haStyle } from "../../../../src/resources/styles";
|
import { haStyle } from "../../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart";
|
import { suggestAddonRestart } from "../../dialogs/suggestAddonRestart";
|
||||||
@@ -38,11 +37,13 @@ interface NetworkItemInput extends PaperInputElement {
|
|||||||
class HassioAddonNetwork extends LitElement {
|
class HassioAddonNetwork extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public addon!: HassioAddonDetails;
|
@property({ attribute: false }) public addon!: HassioAddonDetails;
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
@state() private _error?: string;
|
||||||
|
|
||||||
@internalProperty() private _config?: NetworkItem[];
|
@state() private _config?: NetworkItem[];
|
||||||
|
|
||||||
public connectedCallback(): void {
|
public connectedCallback(): void {
|
||||||
super.connectedCallback();
|
super.connectedCallback();
|
||||||
@@ -55,43 +56,57 @@ class HassioAddonNetwork extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
<ha-card header="Network">
|
<ha-card
|
||||||
|
.header=${this.supervisor.localize(
|
||||||
|
"addon.configuration.network.header"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
|
${this._error ? html` <div class="errors">${this._error}</div> ` : ""}
|
||||||
|
|
||||||
<table>
|
<table>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr>
|
<tr>
|
||||||
<th>Container</th>
|
<th>
|
||||||
<th>Host</th>
|
${this.supervisor.localize(
|
||||||
<th>Description</th>
|
"addon.configuration.network.container"
|
||||||
|
)}
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
${this.supervisor.localize(
|
||||||
|
"addon.configuration.network.host"
|
||||||
|
)}
|
||||||
|
</th>
|
||||||
|
<th>${this.supervisor.localize("common.description")}</th>
|
||||||
</tr>
|
</tr>
|
||||||
${this._config!.map((item) => {
|
${this._config!.map(
|
||||||
return html`
|
(item) => html`
|
||||||
<tr>
|
<tr>
|
||||||
<td>${item.container}</td>
|
<td>${item.container}</td>
|
||||||
<td>
|
<td>
|
||||||
<paper-input
|
<paper-input
|
||||||
@value-changed=${this._configChanged}
|
@value-changed=${this._configChanged}
|
||||||
placeholder="disabled"
|
placeholder="${this.supervisor.localize(
|
||||||
|
"addon.configuration.network.disabled"
|
||||||
|
)}"
|
||||||
.value=${item.host ? String(item.host) : ""}
|
.value=${item.host ? String(item.host) : ""}
|
||||||
.container=${item.container}
|
.container=${item.container}
|
||||||
no-label-float
|
no-label-float
|
||||||
></paper-input>
|
></paper-input>
|
||||||
</td>
|
</td>
|
||||||
<td>${item.description}</td>
|
<td>${this._computeDescription(item)}</td>
|
||||||
</tr>
|
</tr>
|
||||||
`;
|
`
|
||||||
})}
|
)}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
<ha-progress-button class="warning" @click=${this._resetTapped}>
|
<ha-progress-button class="warning" @click=${this._resetTapped}>
|
||||||
Reset to defaults
|
${this.supervisor.localize("common.reset_defaults")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
<ha-progress-button @click=${this._saveTapped}>
|
<ha-progress-button @click=${this._saveTapped}>
|
||||||
Save
|
${this.supervisor.localize("common.save")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
@@ -105,16 +120,20 @@ class HassioAddonNetwork extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private _computeDescription = (item: NetworkItem): string =>
|
||||||
|
this.addon.translations[this.hass.language]?.network?.[item.container]
|
||||||
|
?.description ||
|
||||||
|
this.addon.translations.en?.network?.[item.container]?.description ||
|
||||||
|
item.description;
|
||||||
|
|
||||||
private _setNetworkConfig(): void {
|
private _setNetworkConfig(): void {
|
||||||
const network = this.addon.network || {};
|
const network = this.addon.network || {};
|
||||||
const description = this.addon.network_description || {};
|
const description = this.addon.network_description || {};
|
||||||
const items: NetworkItem[] = Object.keys(network).map((key) => {
|
const items: NetworkItem[] = Object.keys(network).map((key) => ({
|
||||||
return {
|
container: key,
|
||||||
container: key,
|
host: network[key],
|
||||||
host: network[key],
|
description: description[key],
|
||||||
description: description[key],
|
}));
|
||||||
};
|
|
||||||
});
|
|
||||||
this._config = items.sort((a, b) => (a.container > b.container ? 1 : -1));
|
this._config = items.sort((a, b) => (a.container > b.container ? 1 : -1));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -147,12 +166,14 @@ class HassioAddonNetwork extends LitElement {
|
|||||||
};
|
};
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
if (this.addon?.state === "started") {
|
if (this.addon?.state === "started") {
|
||||||
await suggestAddonRestart(this, this.hass, this.addon);
|
await suggestAddonRestart(this, this.hass, this.supervisor, this.addon);
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to set addon network configuration, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.failed_to_reset",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
button.progress = false;
|
button.progress = false;
|
||||||
@@ -181,17 +202,19 @@ class HassioAddonNetwork extends LitElement {
|
|||||||
};
|
};
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
if (this.addon?.state === "started") {
|
if (this.addon?.state === "started") {
|
||||||
await suggestAddonRestart(this, this.hass, this.addon);
|
await suggestAddonRestart(this, this.hass, this.supervisor, this.addon);
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to set addon network configuration, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.failed_to_save",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
button.progress = false;
|
button.progress = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
|
@@ -1,13 +1,5 @@
|
|||||||
import {
|
import "../../../../src/components/ha-card";
|
||||||
css,
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../../src/components/ha-circular-progress";
|
import "../../../../src/components/ha-circular-progress";
|
||||||
import "../../../../src/components/ha-markdown";
|
import "../../../../src/components/ha-markdown";
|
||||||
import {
|
import {
|
||||||
@@ -19,16 +11,20 @@ import "../../../../src/layouts/hass-loading-screen";
|
|||||||
import { haStyle } from "../../../../src/resources/styles";
|
import { haStyle } from "../../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
import { hassioStyle } from "../../resources/hassio-style";
|
import { hassioStyle } from "../../resources/hassio-style";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
|
import { customElement, property, state } from "lit/decorators";
|
||||||
|
|
||||||
@customElement("hassio-addon-documentation-tab")
|
@customElement("hassio-addon-documentation-tab")
|
||||||
class HassioAddonDocumentationDashboard extends LitElement {
|
class HassioAddonDocumentationDashboard extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public addon?: HassioAddonDetails;
|
@property({ attribute: false }) public addon?: HassioAddonDetails;
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
@state() private _error?: string;
|
||||||
|
|
||||||
@internalProperty() private _content?: string;
|
@state() private _content?: string;
|
||||||
|
|
||||||
public async connectedCallback(): Promise<void> {
|
public async connectedCallback(): Promise<void> {
|
||||||
super.connectedCallback();
|
super.connectedCallback();
|
||||||
@@ -53,7 +49,7 @@ class HassioAddonDocumentationDashboard extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
@@ -81,9 +77,11 @@ class HassioAddonDocumentationDashboard extends LitElement {
|
|||||||
this.addon!.slug
|
this.addon!.slug
|
||||||
);
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to get addon documentation, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.documentation.get_logs",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -4,16 +4,8 @@ import {
|
|||||||
mdiInformationVariant,
|
mdiInformationVariant,
|
||||||
mdiMathLog,
|
mdiMathLog,
|
||||||
} from "@mdi/js";
|
} from "@mdi/js";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property, state } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../src/common/dom/fire_event";
|
||||||
import { navigate } from "../../../src/common/navigate";
|
import { navigate } from "../../../src/common/navigate";
|
||||||
@@ -21,6 +13,7 @@ import { extractSearchParam } from "../../../src/common/url/search-params";
|
|||||||
import "../../../src/components/ha-circular-progress";
|
import "../../../src/components/ha-circular-progress";
|
||||||
import {
|
import {
|
||||||
fetchHassioAddonInfo,
|
fetchHassioAddonInfo,
|
||||||
|
fetchHassioAddonsInfo,
|
||||||
HassioAddonDetails,
|
HassioAddonDetails,
|
||||||
} from "../../../src/data/hassio/addon";
|
} from "../../../src/data/hassio/addon";
|
||||||
import { extractApiErrorMessage } from "../../../src/data/hassio/common";
|
import { extractApiErrorMessage } from "../../../src/data/hassio/common";
|
||||||
@@ -51,7 +44,7 @@ class HassioAddonDashboard extends LitElement {
|
|||||||
|
|
||||||
@property({ type: Boolean }) public narrow!: boolean;
|
@property({ type: Boolean }) public narrow!: boolean;
|
||||||
|
|
||||||
@internalProperty() _error?: string;
|
@state() _error?: string;
|
||||||
|
|
||||||
private _computeTail = memoizeOne((route: Route) => {
|
private _computeTail = memoizeOne((route: Route) => {
|
||||||
const dividerPos = route.path.indexOf("/", 1);
|
const dividerPos = route.path.indexOf("/", 1);
|
||||||
@@ -79,7 +72,7 @@ class HassioAddonDashboard extends LitElement {
|
|||||||
|
|
||||||
const addonTabs: PageNavigation[] = [
|
const addonTabs: PageNavigation[] = [
|
||||||
{
|
{
|
||||||
name: "Info",
|
translationKey: "addon.panel.info",
|
||||||
path: `/hassio/addon/${this.addon.slug}/info`,
|
path: `/hassio/addon/${this.addon.slug}/info`,
|
||||||
iconPath: mdiInformationVariant,
|
iconPath: mdiInformationVariant,
|
||||||
},
|
},
|
||||||
@@ -87,7 +80,7 @@ class HassioAddonDashboard extends LitElement {
|
|||||||
|
|
||||||
if (this.addon.documentation) {
|
if (this.addon.documentation) {
|
||||||
addonTabs.push({
|
addonTabs.push({
|
||||||
name: "Documentation",
|
translationKey: "addon.panel.documentation",
|
||||||
path: `/hassio/addon/${this.addon.slug}/documentation`,
|
path: `/hassio/addon/${this.addon.slug}/documentation`,
|
||||||
iconPath: mdiFileDocument,
|
iconPath: mdiFileDocument,
|
||||||
});
|
});
|
||||||
@@ -96,12 +89,12 @@ class HassioAddonDashboard extends LitElement {
|
|||||||
if (this.addon.version) {
|
if (this.addon.version) {
|
||||||
addonTabs.push(
|
addonTabs.push(
|
||||||
{
|
{
|
||||||
name: "Configuration",
|
translationKey: "addon.panel.configuration",
|
||||||
path: `/hassio/addon/${this.addon.slug}/config`,
|
path: `/hassio/addon/${this.addon.slug}/config`,
|
||||||
iconPath: mdiCogs,
|
iconPath: mdiCogs,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "Log",
|
translationKey: "addon.panel.log",
|
||||||
path: `/hassio/addon/${this.addon.slug}/logs`,
|
path: `/hassio/addon/${this.addon.slug}/logs`,
|
||||||
iconPath: mdiMathLog,
|
iconPath: mdiMathLog,
|
||||||
}
|
}
|
||||||
@@ -113,11 +106,12 @@ class HassioAddonDashboard extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
<hass-tabs-subpage
|
<hass-tabs-subpage
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.localizeFunc=${this.supervisor.localize}
|
||||||
.narrow=${this.narrow}
|
.narrow=${this.narrow}
|
||||||
.backPath=${this.addon.version ? "/hassio/dashboard" : "/hassio/store"}
|
.backPath=${this.addon.version ? "/hassio/dashboard" : "/hassio/store"}
|
||||||
.route=${route}
|
.route=${route}
|
||||||
hassio
|
|
||||||
.tabs=${addonTabs}
|
.tabs=${addonTabs}
|
||||||
|
supervisor
|
||||||
>
|
>
|
||||||
<span slot="header">${this.addon.name}</span>
|
<span slot="header">${this.addon.name}</span>
|
||||||
<hassio-addon-router
|
<hassio-addon-router
|
||||||
@@ -131,7 +125,7 @@ class HassioAddonDashboard extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
@@ -172,15 +166,27 @@ class HassioAddonDashboard extends LitElement {
|
|||||||
|
|
||||||
protected async firstUpdated(): Promise<void> {
|
protected async firstUpdated(): Promise<void> {
|
||||||
if (this.route.path === "") {
|
if (this.route.path === "") {
|
||||||
const addon = extractSearchParam("addon");
|
const requestedAddon = extractSearchParam("addon");
|
||||||
if (addon) {
|
if (requestedAddon) {
|
||||||
navigate(this, `/hassio/addon/${addon}`, true);
|
const addonsInfo = await fetchHassioAddonsInfo(this.hass);
|
||||||
|
const validAddon = addonsInfo.addons.some(
|
||||||
|
(addon) => addon.slug === requestedAddon
|
||||||
|
);
|
||||||
|
if (!validAddon) {
|
||||||
|
this._error = this.supervisor.localize("my.error_addon_not_found");
|
||||||
|
} else {
|
||||||
|
navigate(`/hassio/addon/${requestedAddon}`, { replace: true });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this.addEventListener("hass-api-called", (ev) => this._apiCalled(ev));
|
this.addEventListener("hass-api-called", (ev) => this._apiCalled(ev));
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _apiCalled(ev): Promise<void> {
|
private async _apiCalled(ev): Promise<void> {
|
||||||
|
if (!ev.detail.success) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const pathSplit: string[] = ev.detail.path?.split("/");
|
const pathSplit: string[] = ev.detail.path?.split("/");
|
||||||
|
|
||||||
if (!pathSplit || pathSplit.length === 0) {
|
if (!pathSplit || pathSplit.length === 0) {
|
||||||
@@ -190,8 +196,8 @@ class HassioAddonDashboard extends LitElement {
|
|||||||
const path: string = pathSplit[pathSplit.length - 1];
|
const path: string = pathSplit[pathSplit.length - 1];
|
||||||
|
|
||||||
if (["uninstall", "install", "update", "start", "stop"].includes(path)) {
|
if (["uninstall", "install", "update", "start", "stop"].includes(path)) {
|
||||||
fireEvent(this, "supervisor-colllection-refresh", {
|
fireEvent(this, "supervisor-collection-refresh", {
|
||||||
colllection: "supervisor",
|
collection: "supervisor",
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { customElement, property } from "lit-element";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { HassioAddonDetails } from "../../../src/data/hassio/addon";
|
import { HassioAddonDetails } from "../../../src/data/hassio/addon";
|
||||||
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import {
|
import {
|
||||||
|
@@ -1,12 +1,5 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../../src/components/ha-circular-progress";
|
import "../../../../src/components/ha-circular-progress";
|
||||||
import { HassioAddonDetails } from "../../../../src/data/hassio/addon";
|
import { HassioAddonDetails } from "../../../../src/data/hassio/addon";
|
||||||
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
@@ -42,7 +35,7 @@ class HassioAddonInfoDashboard extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
|
@@ -14,17 +14,9 @@ import {
|
|||||||
mdiPound,
|
mdiPound,
|
||||||
mdiShield,
|
mdiShield,
|
||||||
} from "@mdi/js";
|
} from "@mdi/js";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property, state } from "lit/decorators";
|
||||||
CSSResult,
|
import { classMap } from "lit/directives/class-map";
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { classMap } from "lit-html/directives/class-map";
|
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { atLeastVersion } from "../../../../src/common/config/version";
|
import { atLeastVersion } from "../../../../src/common/config/version";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
@@ -50,6 +42,7 @@ import {
|
|||||||
startHassioAddon,
|
startHassioAddon,
|
||||||
stopHassioAddon,
|
stopHassioAddon,
|
||||||
uninstallHassioAddon,
|
uninstallHassioAddon,
|
||||||
|
updateHassioAddon,
|
||||||
validateHassioAddonOption,
|
validateHassioAddonOption,
|
||||||
} from "../../../../src/data/hassio/addon";
|
} from "../../../../src/data/hassio/addon";
|
||||||
import {
|
import {
|
||||||
@@ -68,8 +61,8 @@ import { HomeAssistant } from "../../../../src/types";
|
|||||||
import { bytesToString } from "../../../../src/util/bytes-to-string";
|
import { bytesToString } from "../../../../src/util/bytes-to-string";
|
||||||
import "../../components/hassio-card-content";
|
import "../../components/hassio-card-content";
|
||||||
import "../../components/supervisor-metric";
|
import "../../components/supervisor-metric";
|
||||||
import { showDialogSupervisorAddonUpdate } from "../../dialogs/addon/show-dialog-addon-update";
|
|
||||||
import { showHassioMarkdownDialog } from "../../dialogs/markdown/show-dialog-hassio-markdown";
|
import { showHassioMarkdownDialog } from "../../dialogs/markdown/show-dialog-hassio-markdown";
|
||||||
|
import { showDialogSupervisorUpdate } from "../../dialogs/update/show-dialog-update";
|
||||||
import { hassioStyle } from "../../resources/hassio-style";
|
import { hassioStyle } from "../../resources/hassio-style";
|
||||||
import { addonArchIsSupported } from "../../util/addon";
|
import { addonArchIsSupported } from "../../util/addon";
|
||||||
|
|
||||||
@@ -79,63 +72,6 @@ const STAGE_ICON = {
|
|||||||
deprecated: mdiExclamationThick,
|
deprecated: mdiExclamationThick,
|
||||||
};
|
};
|
||||||
|
|
||||||
const PERMIS_DESC = {
|
|
||||||
stage: {
|
|
||||||
title: "Add-on Stage",
|
|
||||||
description: `Add-ons can have one of three stages:\n\n<ha-svg-icon path="${STAGE_ICON.stable}"></ha-svg-icon> **Stable**: These are add-ons ready to be used in production.\n\n<ha-svg-icon path="${STAGE_ICON.experimental}"></ha-svg-icon> **Experimental**: These may contain bugs, and may be unfinished.\n\n<ha-svg-icon path="${STAGE_ICON.deprecated}"></ha-svg-icon> **Deprecated**: These add-ons will no longer receive any updates.`,
|
|
||||||
},
|
|
||||||
rating: {
|
|
||||||
title: "Add-on Security Rating",
|
|
||||||
description:
|
|
||||||
"Home Assistant provides a security rating to each of the add-ons, which indicates the risks involved when using this add-on. The more access an add-on requires on your system, the lower the score, thus raising the possible security risks.\n\nA score is on a scale from 1 to 6. Where 1 is the lowest score (considered the most insecure and highest risk) and a score of 6 is the highest score (considered the most secure and lowest risk).",
|
|
||||||
},
|
|
||||||
host_network: {
|
|
||||||
title: "Host Network",
|
|
||||||
description:
|
|
||||||
"Add-ons usually run in their own isolated network layer, which prevents them from accessing the network of the host operating system. In some cases, this network isolation can limit add-ons in providing their services and therefore, the isolation can be lifted by the add-on author, giving the add-on full access to the network capabilities of the host machine. This gives the add-on more networking capabilities but lowers the security, hence, the security rating of the add-on will be lowered when this option is used by the add-on.",
|
|
||||||
},
|
|
||||||
homeassistant_api: {
|
|
||||||
title: "Home Assistant API Access",
|
|
||||||
description:
|
|
||||||
"This add-on is allowed to access your running Home Assistant instance directly via the Home Assistant API. This mode handles authentication for the add-on as well, which enables an add-on to interact with Home Assistant without the need for additional authentication tokens.",
|
|
||||||
},
|
|
||||||
full_access: {
|
|
||||||
title: "Full Hardware Access",
|
|
||||||
description:
|
|
||||||
"This add-on is given full access to the hardware of your system, by request of the add-on author. Access is comparable to the privileged mode in Docker. Since this opens up possible security risks, this feature impacts the add-on security score negatively.\n\nThis level of access is not granted automatically and needs to be confirmed by you. To do this, you need to disable the protection mode on the add-on manually. Only disable the protection mode if you know, need AND trust the source of this add-on.",
|
|
||||||
},
|
|
||||||
hassio_api: {
|
|
||||||
title: "Supervisor API Access",
|
|
||||||
description:
|
|
||||||
"The add-on was given access to the Supervisor API, by request of the add-on author. By default, the add-on can access general version information of your system. When the add-on requests 'manager' or 'admin' level access to the API, it will gain access to control multiple parts of your Home Assistant system. This permission is indicated by this badge and will impact the security score of the addon negatively.",
|
|
||||||
},
|
|
||||||
docker_api: {
|
|
||||||
title: "Full Docker Access",
|
|
||||||
description:
|
|
||||||
"The add-on author has requested the add-on to have management access to the Docker instance running on your system. This mode gives the add-on full access and control to your entire Home Assistant system, which adds security risks, and could damage your system when misused. Therefore, this feature impacts the add-on security score negatively.\n\nThis level of access is not granted automatically and needs to be confirmed by you. To do this, you need to disable the protection mode on the add-on manually. Only disable the protection mode if you know, need AND trust the source of this add-on.",
|
|
||||||
},
|
|
||||||
host_pid: {
|
|
||||||
title: "Host Processes Namespace",
|
|
||||||
description:
|
|
||||||
"Usually, the processes the add-on runs, are isolated from all other system processes. The add-on author has requested the add-on to have access to the system processes running on the host system instance, and allow the add-on to spawn processes on the host system as well. This mode gives the add-on full access and control to your entire Home Assistant system, which adds security risks, and could damage your system when misused. Therefore, this feature impacts the add-on security score negatively.\n\nThis level of access is not granted automatically and needs to be confirmed by you. To do this, you need to disable the protection mode on the add-on manually. Only disable the protection mode if you know, need AND trust the source of this add-on.",
|
|
||||||
},
|
|
||||||
apparmor: {
|
|
||||||
title: "AppArmor",
|
|
||||||
description:
|
|
||||||
"AppArmor ('Application Armor') is a Linux kernel security module that restricts add-ons capabilities like network access, raw socket access, and permission to read, write, or execute specific files.\n\nAdd-on authors can provide their security profiles, optimized for the add-on, or request it to be disabled. If AppArmor is disabled, it will raise security risks and therefore, has a negative impact on the security score of the add-on.",
|
|
||||||
},
|
|
||||||
auth_api: {
|
|
||||||
title: "Home Assistant Authentication",
|
|
||||||
description:
|
|
||||||
"An add-on can authenticate users against Home Assistant, allowing add-ons to give users the possibility to log into applications running inside add-ons, using their Home Assistant username/password. This badge indicates if the add-on author requests this capability.",
|
|
||||||
},
|
|
||||||
ingress: {
|
|
||||||
title: "Ingress",
|
|
||||||
description:
|
|
||||||
"This add-on is using Ingress to embed its interface securely into Home Assistant.",
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
@customElement("hassio-addon-info")
|
@customElement("hassio-addon-info")
|
||||||
class HassioAddonInfo extends LitElement {
|
class HassioAddonInfo extends LitElement {
|
||||||
@property({ type: Boolean }) public narrow!: boolean;
|
@property({ type: Boolean }) public narrow!: boolean;
|
||||||
@@ -146,9 +82,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
|
|
||||||
@property({ attribute: false }) public supervisor!: Supervisor;
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@internalProperty() private _metrics?: HassioStats;
|
@state() private _metrics?: HassioStats;
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
@state() private _error?: string;
|
||||||
|
|
||||||
private _addonStoreInfo = memoizeOne(
|
private _addonStoreInfo = memoizeOne(
|
||||||
(slug: string, storeAddons: StoreAddon[]) =>
|
(slug: string, storeAddons: StoreAddon[]) =>
|
||||||
@@ -162,11 +98,11 @@ class HassioAddonInfo extends LitElement {
|
|||||||
: undefined;
|
: undefined;
|
||||||
const metrics = [
|
const metrics = [
|
||||||
{
|
{
|
||||||
description: "Add-on CPU Usage",
|
description: this.supervisor.localize("addon.dashboard.cpu_usage"),
|
||||||
value: this._metrics?.cpu_percent,
|
value: this._metrics?.cpu_percent,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
description: "Add-on RAM Usage",
|
description: this.supervisor.localize("addon.dashboard.ram_usage"),
|
||||||
value: this._metrics?.memory_percent,
|
value: this._metrics?.memory_percent,
|
||||||
tooltip: `${bytesToString(this._metrics?.memory_usage)}/${bytesToString(
|
tooltip: `${bytesToString(this._metrics?.memory_usage)}/${bytesToString(
|
||||||
this._metrics?.memory_limit
|
this._metrics?.memory_limit
|
||||||
@@ -176,14 +112,28 @@ class HassioAddonInfo extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
${this.addon.update_available
|
${this.addon.update_available
|
||||||
? html`
|
? html`
|
||||||
<ha-card header="Update available! 🎉">
|
<ha-card
|
||||||
|
.header="${this.supervisor.localize(
|
||||||
|
"common.update_available",
|
||||||
|
"count",
|
||||||
|
1
|
||||||
|
)}🎉"
|
||||||
|
>
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
<hassio-card-content
|
<hassio-card-content
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.title="${this.addon.name} ${this.addon
|
.title="${this.supervisor.localize(
|
||||||
.version_latest} is available"
|
"addon.dashboard.new_update_available",
|
||||||
.description="You are currently running version ${this.addon
|
"name",
|
||||||
.version}"
|
this.addon.name,
|
||||||
|
"version",
|
||||||
|
this.addon.version_latest
|
||||||
|
)}"
|
||||||
|
.description="${this.supervisor.localize(
|
||||||
|
"common.running_version",
|
||||||
|
"version",
|
||||||
|
this.addon.version
|
||||||
|
)}"
|
||||||
icon=${mdiArrowUpBoldCircle}
|
icon=${mdiArrowUpBoldCircle}
|
||||||
iconClass="update"
|
iconClass="update"
|
||||||
></hassio-card-content>
|
></hassio-card-content>
|
||||||
@@ -194,32 +144,35 @@ class HassioAddonInfo extends LitElement {
|
|||||||
)
|
)
|
||||||
? html`
|
? html`
|
||||||
<p class="warning">
|
<p class="warning">
|
||||||
This add-on is not compatible with the processor of
|
${this.supervisor.localize(
|
||||||
your device or the operating system you have installed
|
"addon.dashboard.not_available_arch"
|
||||||
on your device.
|
)}
|
||||||
</p>
|
</p>
|
||||||
`
|
`
|
||||||
: html`
|
: html`
|
||||||
<p class="warning">
|
<p class="warning">
|
||||||
You are running Home Assistant
|
${this.supervisor.localize(
|
||||||
${this.supervisor.core.version}, to update to this
|
"addon.dashboard.not_available_arch",
|
||||||
version of the add-on you need at least version
|
"core_version_installed",
|
||||||
${addonStoreInfo.homeassistant} of Home Assistant
|
this.supervisor.core.version,
|
||||||
|
"core_version_needed",
|
||||||
|
addonStoreInfo.homeassistant
|
||||||
|
)}
|
||||||
</p>
|
</p>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
<mwc-button @click=${this._updateClicked}>
|
|
||||||
Update
|
|
||||||
</mwc-button>
|
|
||||||
${this.addon.changelog
|
${this.addon.changelog
|
||||||
? html`
|
? html`
|
||||||
<mwc-button @click=${this._openChangelog}>
|
<mwc-button @click=${this._openChangelog}>
|
||||||
Changelog
|
${this.supervisor.localize("addon.dashboard.changelog")}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
`
|
`
|
||||||
: ""}
|
: html`<span></span>`}
|
||||||
|
<mwc-button @click=${this._updateClicked}>
|
||||||
|
${this.supervisor.localize("common.update")}
|
||||||
|
</mwc-button>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
`
|
`
|
||||||
@@ -227,12 +180,19 @@ class HassioAddonInfo extends LitElement {
|
|||||||
${!this.addon.protected
|
${!this.addon.protected
|
||||||
? html`
|
? html`
|
||||||
<ha-card class="warning">
|
<ha-card class="warning">
|
||||||
<h1 class="card-header">Warning: Protection mode is disabled!</h1>
|
<h1 class="card-header">${this.supervisor.localize(
|
||||||
|
"addon.dashboard.protection_mode.title"
|
||||||
|
)}
|
||||||
|
</h1>
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
Protection mode on this add-on is disabled! This gives the add-on full access to the entire system, which adds security risks, and could damage your system when used incorrectly. Only disable the protection mode if you know, need AND trust the source of this add-on.
|
${this.supervisor.localize("addon.dashboard.protection_mode.content")}
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions protection-enable">
|
<div class="card-actions protection-enable">
|
||||||
<mwc-button @click=${this._protectionToggled}>Enable Protection mode</mwc-button>
|
<mwc-button @click=${this._protectionToggled}>
|
||||||
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.protection_mode.enable"
|
||||||
|
)}
|
||||||
|
</mwc-button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
@@ -249,14 +209,18 @@ class HassioAddonInfo extends LitElement {
|
|||||||
${this._computeIsRunning
|
${this._computeIsRunning
|
||||||
? html`
|
? html`
|
||||||
<ha-svg-icon
|
<ha-svg-icon
|
||||||
title="Add-on is running"
|
.title=${this.supervisor.localize(
|
||||||
|
"dashboard.addon_running"
|
||||||
|
)}
|
||||||
class="running"
|
class="running"
|
||||||
.path=${mdiCircle}
|
.path=${mdiCircle}
|
||||||
></ha-svg-icon>
|
></ha-svg-icon>
|
||||||
`
|
`
|
||||||
: html`
|
: html`
|
||||||
<ha-svg-icon
|
<ha-svg-icon
|
||||||
title="Add-on is stopped"
|
.title=${this.supervisor.localize(
|
||||||
|
"dashboard.addon_stopped"
|
||||||
|
)}
|
||||||
class="stopped"
|
class="stopped"
|
||||||
.path=${mdiCircle}
|
.path=${mdiCircle}
|
||||||
></ha-svg-icon>
|
></ha-svg-icon>
|
||||||
@@ -270,21 +234,29 @@ class HassioAddonInfo extends LitElement {
|
|||||||
? html`
|
? html`
|
||||||
Current version: ${this.addon.version}
|
Current version: ${this.addon.version}
|
||||||
<div class="changelog" @click=${this._openChangelog}>
|
<div class="changelog" @click=${this._openChangelog}>
|
||||||
(<span class="changelog-link">changelog</span>)
|
(<span class="changelog-link"
|
||||||
|
>${this.supervisor.localize(
|
||||||
|
"addon.dashboard.changelog"
|
||||||
|
)}</span
|
||||||
|
>)
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
: html`<span class="changelog-link" @click=${this._openChangelog}
|
: html`<span class="changelog-link" @click=${this._openChangelog}
|
||||||
>Changelog</span
|
>${this.supervisor.localize(
|
||||||
|
"addon.dashboard.changelog"
|
||||||
|
)}</span
|
||||||
>`}
|
>`}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="description light-color">
|
<div class="description light-color">
|
||||||
${this.addon.description}.<br />
|
${this.addon.description}.<br />
|
||||||
Visit
|
${this.supervisor.localize(
|
||||||
<a href="${this.addon.url!}" target="_blank" rel="noreferrer">
|
"addon.dashboard.visit_addon_page",
|
||||||
${this.addon.name} page</a
|
"name",
|
||||||
>
|
html`<a href="${this.addon.url!}" target="_blank" rel="noreferrer"
|
||||||
for details.
|
>${this.addon.name}</a
|
||||||
|
>`
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
<div class="addon-container">
|
<div class="addon-container">
|
||||||
<div>
|
<div>
|
||||||
@@ -305,7 +277,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
})}
|
})}
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="stage"
|
id="stage"
|
||||||
label="stage"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.stage"
|
||||||
|
)}
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon
|
<ha-svg-icon
|
||||||
@@ -331,7 +305,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
<ha-label-badge
|
<ha-label-badge
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="host_network"
|
id="host_network"
|
||||||
label="host"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.host"
|
||||||
|
)}
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiNetwork}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiNetwork}></ha-svg-icon>
|
||||||
@@ -343,7 +319,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
<ha-label-badge
|
<ha-label-badge
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="full_access"
|
id="full_access"
|
||||||
label="hardware"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.hardware"
|
||||||
|
)}
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiChip}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiChip}></ha-svg-icon>
|
||||||
@@ -355,7 +333,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
<ha-label-badge
|
<ha-label-badge
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="homeassistant_api"
|
id="homeassistant_api"
|
||||||
label="hass"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.hass"
|
||||||
|
)}
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon>
|
||||||
@@ -367,8 +347,12 @@ class HassioAddonInfo extends LitElement {
|
|||||||
<ha-label-badge
|
<ha-label-badge
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="hassio_api"
|
id="hassio_api"
|
||||||
label="hassio"
|
.label=${this.supervisor.localize(
|
||||||
.description=${this.addon.hassio_role}
|
"addon.dashboard.capability.label.hassio"
|
||||||
|
)}
|
||||||
|
.description=${this.supervisor.localize(
|
||||||
|
`addon.dashboard.capability.role.${this.addon.hassio_role}`
|
||||||
|
) || this.addon.hassio_role}
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon>
|
||||||
</ha-label-badge>
|
</ha-label-badge>
|
||||||
@@ -379,7 +363,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
<ha-label-badge
|
<ha-label-badge
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="docker_api"
|
id="docker_api"
|
||||||
label="docker"
|
.label=".${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.docker"
|
||||||
|
)}"
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiDocker}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiDocker}></ha-svg-icon>
|
||||||
@@ -391,7 +377,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
<ha-label-badge
|
<ha-label-badge
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="host_pid"
|
id="host_pid"
|
||||||
label="host pid"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.host_pid"
|
||||||
|
)}
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiPound}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiPound}></ha-svg-icon>
|
||||||
@@ -404,7 +392,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
class=${this._computeApparmorClassName}
|
class=${this._computeApparmorClassName}
|
||||||
id="apparmor"
|
id="apparmor"
|
||||||
label="apparmor"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.apparmor"
|
||||||
|
)}
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiShield}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiShield}></ha-svg-icon>
|
||||||
@@ -416,7 +406,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
<ha-label-badge
|
<ha-label-badge
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="auth_api"
|
id="auth_api"
|
||||||
label="auth"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.auth"
|
||||||
|
)}
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiKey}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiKey}></ha-svg-icon>
|
||||||
@@ -428,7 +420,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
<ha-label-badge
|
<ha-label-badge
|
||||||
@click=${this._showMoreInfo}
|
@click=${this._showMoreInfo}
|
||||||
id="ingress"
|
id="ingress"
|
||||||
label="ingress"
|
.label=${this.supervisor.localize(
|
||||||
|
"addon.dashboard.capability.label.ingress"
|
||||||
|
)}
|
||||||
description=""
|
description=""
|
||||||
>
|
>
|
||||||
<ha-svg-icon
|
<ha-svg-icon
|
||||||
@@ -449,10 +443,14 @@ class HassioAddonInfo extends LitElement {
|
|||||||
>
|
>
|
||||||
<ha-settings-row ?three-line=${this.narrow}>
|
<ha-settings-row ?three-line=${this.narrow}>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Start on boot
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.option.boot.title"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
Make the add-on start during a system boot
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.option.boot.description"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._startOnBootToggled}
|
@change=${this._startOnBootToggled}
|
||||||
@@ -465,10 +463,14 @@ class HassioAddonInfo extends LitElement {
|
|||||||
? html`
|
? html`
|
||||||
<ha-settings-row ?three-line=${this.narrow}>
|
<ha-settings-row ?three-line=${this.narrow}>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Watchdog
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.option.watchdog.title"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
This will start the add-on if it crashes
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.option.watchdog.description"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._watchdogToggled}
|
@change=${this._watchdogToggled}
|
||||||
@@ -483,11 +485,14 @@ class HassioAddonInfo extends LitElement {
|
|||||||
? html`
|
? html`
|
||||||
<ha-settings-row ?three-line=${this.narrow}>
|
<ha-settings-row ?three-line=${this.narrow}>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Auto update
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.option.auto_update.title"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
Auto update the add-on when there is a new
|
${this.supervisor.localize(
|
||||||
version available
|
"addon.dashboard.option.auto_update.description"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._autoUpdateToggled}
|
@change=${this._autoUpdateToggled}
|
||||||
@@ -497,21 +502,22 @@ class HassioAddonInfo extends LitElement {
|
|||||||
</ha-settings-row>
|
</ha-settings-row>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
${this.addon.ingress
|
${!this._computeCannotIngressSidebar && this.addon.ingress
|
||||||
? html`
|
? html`
|
||||||
<ha-settings-row ?three-line=${this.narrow}>
|
<ha-settings-row ?three-line=${this.narrow}>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Show in sidebar
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.option.ingress_panel.title"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
${this._computeCannotIngressSidebar
|
${this.supervisor.localize(
|
||||||
? "This option requires Home Assistant 0.92 or later."
|
"addon.dashboard.option.ingress_panel.description"
|
||||||
: "Add this add-on to your sidebar"}
|
)}
|
||||||
</span>
|
</span>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._panelToggled}
|
@change=${this._panelToggled}
|
||||||
.checked=${this.addon.ingress_panel}
|
.checked=${this.addon.ingress_panel}
|
||||||
.disabled=${this._computeCannotIngressSidebar}
|
|
||||||
haptic
|
haptic
|
||||||
></ha-switch>
|
></ha-switch>
|
||||||
</ha-settings-row>
|
</ha-settings-row>
|
||||||
@@ -521,10 +527,14 @@ class HassioAddonInfo extends LitElement {
|
|||||||
? html`
|
? html`
|
||||||
<ha-settings-row ?three-line=${this.narrow}>
|
<ha-settings-row ?three-line=${this.narrow}>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Protection mode
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.option.protected.title"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
Blocks elevated system access from the add-on
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.option.protected.description"
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._protectionToggled}
|
@change=${this._protectionToggled}
|
||||||
@@ -542,11 +552,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
${this.addon.state === "started"
|
${this.addon.state === "started"
|
||||||
? html`<ha-settings-row ?three-line=${this.narrow}>
|
? html`<ha-settings-row ?three-line=${this.narrow}>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Hostname
|
${this.supervisor.localize("addon.dashboard.hostname")}
|
||||||
</span>
|
</span>
|
||||||
<code slot="description">
|
<code slot="description"> ${this.addon.hostname} </code>
|
||||||
${this.addon.hostname}
|
|
||||||
</code>
|
|
||||||
</ha-settings-row>
|
</ha-settings-row>
|
||||||
${metrics.map(
|
${metrics.map(
|
||||||
(metric) =>
|
(metric) =>
|
||||||
@@ -569,17 +577,20 @@ class HassioAddonInfo extends LitElement {
|
|||||||
)
|
)
|
||||||
? html`
|
? html`
|
||||||
<p class="warning">
|
<p class="warning">
|
||||||
This add-on is not compatible with the processor of your
|
${this.supervisor.localize(
|
||||||
device or the operating system you have installed on your
|
"addon.dashboard.not_available_arch"
|
||||||
device.
|
)}
|
||||||
</p>
|
</p>
|
||||||
`
|
`
|
||||||
: html`
|
: html`
|
||||||
<p class="warning">
|
<p class="warning">
|
||||||
You are running Home Assistant
|
${this.supervisor.localize(
|
||||||
${this.supervisor.core.version}, to install this add-on you
|
"addon.dashboard.not_available_version",
|
||||||
need at least version ${addonStoreInfo!.homeassistant} of
|
"core_version_installed",
|
||||||
Home Assistant
|
this.supervisor.core.version,
|
||||||
|
"core_version_needed",
|
||||||
|
addonStoreInfo!.homeassistant
|
||||||
|
)}
|
||||||
</p>
|
</p>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
@@ -593,18 +604,18 @@ class HassioAddonInfo extends LitElement {
|
|||||||
class="warning"
|
class="warning"
|
||||||
@click=${this._stopClicked}
|
@click=${this._stopClicked}
|
||||||
>
|
>
|
||||||
Stop
|
${this.supervisor.localize("addon.dashboard.stop")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
class="warning"
|
class="warning"
|
||||||
@click=${this._restartClicked}
|
@click=${this._restartClicked}
|
||||||
>
|
>
|
||||||
Restart
|
${this.supervisor.localize("addon.dashboard.restart")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
`
|
`
|
||||||
: html`
|
: html`
|
||||||
<ha-progress-button @click=${this._startClicked}>
|
<ha-progress-button @click=${this._startClicked}>
|
||||||
Start
|
${this.supervisor.localize("addon.dashboard.start")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
`
|
`
|
||||||
: html`
|
: html`
|
||||||
@@ -612,7 +623,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
.disabled=${!this.addon.available}
|
.disabled=${!this.addon.available}
|
||||||
@click=${this._installClicked}
|
@click=${this._installClicked}
|
||||||
>
|
>
|
||||||
Install
|
${this.supervisor.localize("addon.dashboard.install")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
`}
|
`}
|
||||||
</div>
|
</div>
|
||||||
@@ -627,7 +638,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
rel="noopener"
|
rel="noopener"
|
||||||
>
|
>
|
||||||
<mwc-button>
|
<mwc-button>
|
||||||
Open web UI
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.open_web_ui"
|
||||||
|
)}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
</a>
|
</a>
|
||||||
`
|
`
|
||||||
@@ -635,7 +648,9 @@ class HassioAddonInfo extends LitElement {
|
|||||||
${this._computeShowIngressUI
|
${this._computeShowIngressUI
|
||||||
? html`
|
? html`
|
||||||
<mwc-button @click=${this._openIngress}>
|
<mwc-button @click=${this._openIngress}>
|
||||||
Open web UI
|
${this.supervisor.localize(
|
||||||
|
"addon.dashboard.open_web_ui"
|
||||||
|
)}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
@@ -643,7 +658,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
class="warning"
|
class="warning"
|
||||||
@click=${this._uninstallClicked}
|
@click=${this._uninstallClicked}
|
||||||
>
|
>
|
||||||
Uninstall
|
${this.supervisor.localize("addon.dashboard.uninstall")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
${this.addon.build
|
${this.addon.build
|
||||||
? html`
|
? html`
|
||||||
@@ -652,7 +667,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.path="hassio/addons/${this.addon.slug}/rebuild"
|
.path="hassio/addons/${this.addon.slug}/rebuild"
|
||||||
>
|
>
|
||||||
Rebuild
|
${this.supervisor.localize("addon.dashboard.rebuild")}
|
||||||
</ha-call-api-button>
|
</ha-call-api-button>
|
||||||
`
|
`
|
||||||
: ""}`
|
: ""}`
|
||||||
@@ -712,8 +727,21 @@ class HassioAddonInfo extends LitElement {
|
|||||||
private _showMoreInfo(ev): void {
|
private _showMoreInfo(ev): void {
|
||||||
const id = ev.currentTarget.id;
|
const id = ev.currentTarget.id;
|
||||||
showHassioMarkdownDialog(this, {
|
showHassioMarkdownDialog(this, {
|
||||||
title: PERMIS_DESC[id].title,
|
title: this.supervisor.localize(`addon.dashboard.capability.${id}.title`),
|
||||||
content: PERMIS_DESC[id].description,
|
content:
|
||||||
|
id === "stage"
|
||||||
|
? this.supervisor.localize(
|
||||||
|
`addon.dashboard.capability.${id}.description`,
|
||||||
|
"icon_stable",
|
||||||
|
`<ha-svg-icon path="${STAGE_ICON.stable}"></ha-svg-icon>`,
|
||||||
|
"icon_experimental",
|
||||||
|
`<ha-svg-icon path="${STAGE_ICON.experimental}"></ha-svg-icon>`,
|
||||||
|
"icon_deprecated",
|
||||||
|
`<ha-svg-icon path="${STAGE_ICON.deprecated}"></ha-svg-icon>`
|
||||||
|
)
|
||||||
|
: this.supervisor.localize(
|
||||||
|
`addon.dashboard.capability.${id}.description`
|
||||||
|
),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -733,7 +761,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private _openIngress(): void {
|
private _openIngress(): void {
|
||||||
navigate(this, `/hassio/ingress/${this.addon.slug}`);
|
navigate(`/hassio/ingress/${this.addon.slug}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
private get _computeShowIngressUI(): boolean {
|
private get _computeShowIngressUI(): boolean {
|
||||||
@@ -766,9 +794,11 @@ class HassioAddonInfo extends LitElement {
|
|||||||
};
|
};
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to set addon option, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.failed_to_save",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -786,9 +816,11 @@ class HassioAddonInfo extends LitElement {
|
|||||||
};
|
};
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to set addon option, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.failed_to_save",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -806,9 +838,11 @@ class HassioAddonInfo extends LitElement {
|
|||||||
};
|
};
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to set addon option, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.failed_to_save",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -826,9 +860,11 @@ class HassioAddonInfo extends LitElement {
|
|||||||
};
|
};
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to set addon security option, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.failed_to_save",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -846,9 +882,11 @@ class HassioAddonInfo extends LitElement {
|
|||||||
};
|
};
|
||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to set addon option, ${extractApiErrorMessage(
|
this._error = this.supervisor.localize(
|
||||||
err
|
"addon.failed_to_save",
|
||||||
)}`;
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -859,12 +897,14 @@ class HassioAddonInfo extends LitElement {
|
|||||||
this.addon.slug
|
this.addon.slug
|
||||||
);
|
);
|
||||||
showHassioMarkdownDialog(this, {
|
showHassioMarkdownDialog(this, {
|
||||||
title: "Changelog",
|
title: this.supervisor.localize("addon.dashboard.changelog"),
|
||||||
content,
|
content,
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to get addon changelog",
|
title: this.supervisor.localize(
|
||||||
|
"addon.dashboard.action_error.get_changelog"
|
||||||
|
),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -884,7 +924,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to install addon",
|
title: this.supervisor.localize("addon.dashboard.action_error.install"),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -905,7 +945,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to stop addon",
|
title: this.supervisor.localize("addon.dashboard.action_error.stop"),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -926,7 +966,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to restart addon",
|
title: this.supervisor.localize("addon.dashboard.action_error.restart"),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -934,7 +974,30 @@ class HassioAddonInfo extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async _updateClicked(): Promise<void> {
|
private async _updateClicked(): Promise<void> {
|
||||||
showDialogSupervisorAddonUpdate(this, { addon: this.addon });
|
showDialogSupervisorUpdate(this, {
|
||||||
|
supervisor: this.supervisor,
|
||||||
|
name: this.addon.name,
|
||||||
|
version: this.addon.version_latest,
|
||||||
|
snapshotParams: {
|
||||||
|
name: `addon_${this.addon.slug}_${this.addon.version}`,
|
||||||
|
addons: [this.addon.slug],
|
||||||
|
homeassistant: false,
|
||||||
|
},
|
||||||
|
updateHandler: async () => this._updateAddon(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _updateAddon(): Promise<void> {
|
||||||
|
await updateHassioAddon(this.hass, this.addon.slug);
|
||||||
|
fireEvent(this, "supervisor-collection-refresh", {
|
||||||
|
collection: "addon",
|
||||||
|
});
|
||||||
|
const eventdata = {
|
||||||
|
success: true,
|
||||||
|
response: undefined,
|
||||||
|
path: "update",
|
||||||
|
};
|
||||||
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _startClicked(ev: CustomEvent): Promise<void> {
|
private async _startClicked(ev: CustomEvent): Promise<void> {
|
||||||
@@ -947,11 +1010,15 @@ class HassioAddonInfo extends LitElement {
|
|||||||
);
|
);
|
||||||
if (!validate.valid) {
|
if (!validate.valid) {
|
||||||
await showConfirmationDialog(this, {
|
await showConfirmationDialog(this, {
|
||||||
title: "Failed to start addon - configuration validation failed!",
|
title: this.supervisor.localize(
|
||||||
|
"addon.dashboard.action_error.start_invalid_config"
|
||||||
|
),
|
||||||
text: validate.message.split(" Got ")[0],
|
text: validate.message.split(" Got ")[0],
|
||||||
confirm: () => this._openConfiguration(),
|
confirm: () => this._openConfiguration(),
|
||||||
confirmText: "Go to configuration",
|
confirmText: this.supervisor.localize(
|
||||||
dismissText: "Cancel",
|
"addon.dashboard.action_error.go_to_config"
|
||||||
|
),
|
||||||
|
dismissText: this.supervisor.localize("common.cancel"),
|
||||||
});
|
});
|
||||||
button.progress = false;
|
button.progress = false;
|
||||||
return;
|
return;
|
||||||
@@ -976,7 +1043,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to start addon",
|
title: this.supervisor.localize("addon.dashboard.action_error.start"),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -984,7 +1051,7 @@ class HassioAddonInfo extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private _openConfiguration(): void {
|
private _openConfiguration(): void {
|
||||||
navigate(this, `/hassio/addon/${this.addon.slug}/config`);
|
navigate(`/hassio/addon/${this.addon.slug}/config`);
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _uninstallClicked(ev: CustomEvent): Promise<void> {
|
private async _uninstallClicked(ev: CustomEvent): Promise<void> {
|
||||||
@@ -1014,14 +1081,16 @@ class HassioAddonInfo extends LitElement {
|
|||||||
fireEvent(this, "hass-api-called", eventdata);
|
fireEvent(this, "hass-api-called", eventdata);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to uninstall addon",
|
title: this.supervisor.localize(
|
||||||
|
"addon.dashboard.action_error.uninstall"
|
||||||
|
),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
button.progress = false;
|
button.progress = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
|
@@ -1,14 +1,8 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../../src/components/ha-circular-progress";
|
import "../../../../src/components/ha-circular-progress";
|
||||||
import { HassioAddonDetails } from "../../../../src/data/hassio/addon";
|
import { HassioAddonDetails } from "../../../../src/data/hassio/addon";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import { haStyle } from "../../../../src/resources/styles";
|
import { haStyle } from "../../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
import { hassioStyle } from "../../resources/hassio-style";
|
import { hassioStyle } from "../../resources/hassio-style";
|
||||||
@@ -18,6 +12,8 @@ import "./hassio-addon-logs";
|
|||||||
class HassioAddonLogDashboard extends LitElement {
|
class HassioAddonLogDashboard extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public addon?: HassioAddonDetails;
|
@property({ attribute: false }) public addon?: HassioAddonDetails;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
@@ -28,13 +24,14 @@ class HassioAddonLogDashboard extends LitElement {
|
|||||||
<div class="content">
|
<div class="content">
|
||||||
<hassio-addon-logs
|
<hassio-addon-logs
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.supervisor=${this.supervisor}
|
||||||
.addon=${this.addon}
|
.addon=${this.addon}
|
||||||
></hassio-addon-logs>
|
></hassio-addon-logs>
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
|
@@ -1,20 +1,13 @@
|
|||||||
import "@material/mwc-button";
|
import "@material/mwc-button";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property, state } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
import {
|
import {
|
||||||
fetchHassioAddonLogs,
|
fetchHassioAddonLogs,
|
||||||
HassioAddonDetails,
|
HassioAddonDetails,
|
||||||
} from "../../../../src/data/hassio/addon";
|
} from "../../../../src/data/hassio/addon";
|
||||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import { haStyle } from "../../../../src/resources/styles";
|
import { haStyle } from "../../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
import "../../components/hassio-ansi-to-html";
|
import "../../components/hassio-ansi-to-html";
|
||||||
@@ -24,11 +17,13 @@ import { hassioStyle } from "../../resources/hassio-style";
|
|||||||
class HassioAddonLogs extends LitElement {
|
class HassioAddonLogs extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public addon!: HassioAddonDetails;
|
@property({ attribute: false }) public addon!: HassioAddonDetails;
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
@state() private _error?: string;
|
||||||
|
|
||||||
@internalProperty() private _content?: string;
|
@state() private _content?: string;
|
||||||
|
|
||||||
public async connectedCallback(): Promise<void> {
|
public async connectedCallback(): Promise<void> {
|
||||||
super.connectedCallback();
|
super.connectedCallback();
|
||||||
@@ -48,13 +43,15 @@ class HassioAddonLogs extends LitElement {
|
|||||||
: ""}
|
: ""}
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
<mwc-button @click=${this._refresh}>Refresh</mwc-button>
|
<mwc-button @click=${this._refresh}>
|
||||||
|
${this.supervisor.localize("common.refresh")}
|
||||||
|
</mwc-button>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
@@ -76,7 +73,11 @@ class HassioAddonLogs extends LitElement {
|
|||||||
try {
|
try {
|
||||||
this._content = await fetchHassioAddonLogs(this.hass, this.addon.slug);
|
this._content = await fetchHassioAddonLogs(this.hass, this.addon.slug);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = `Failed to get addon logs, ${extractApiErrorMessage(err)}`;
|
this._error = this.supervisor.localize(
|
||||||
|
"addon.logs.get_logs",
|
||||||
|
"error",
|
||||||
|
extractApiErrorMessage(err)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,12 +1,5 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
|
|
||||||
interface State {
|
interface State {
|
||||||
bold: boolean;
|
bold: boolean;
|
||||||
@@ -25,7 +18,7 @@ class HassioAnsiToHtml extends LitElement {
|
|||||||
return html`${this._parseTextToColoredPre(this.content)}`;
|
return html`${this._parseTextToColoredPre(this.content)}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
pre {
|
pre {
|
||||||
overflow-x: auto;
|
overflow-x: auto;
|
||||||
|
@@ -1,13 +1,6 @@
|
|||||||
import { mdiHelpCircle } from "@mdi/js";
|
import { mdiHelpCircle } from "@mdi/js";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../src/components/ha-relative-time";
|
import "../../../src/components/ha-relative-time";
|
||||||
import "../../../src/components/ha-svg-icon";
|
import "../../../src/components/ha-svg-icon";
|
||||||
import { HomeAssistant } from "../../../src/types";
|
import { HomeAssistant } from "../../../src/types";
|
||||||
@@ -44,7 +37,7 @@ class HassioCardContent extends LitElement {
|
|||||||
${this.iconImage
|
${this.iconImage
|
||||||
? html`
|
? html`
|
||||||
<div class="icon_image ${this.iconClass}">
|
<div class="icon_image ${this.iconClass}">
|
||||||
<img src="${this.iconImage}" title="${this.iconTitle}" />
|
<img src="${this.iconImage}" .title=${this.iconTitle} />
|
||||||
<div></div>
|
<div></div>
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
@@ -56,13 +49,13 @@ class HassioCardContent extends LitElement {
|
|||||||
></ha-svg-icon>
|
></ha-svg-icon>
|
||||||
`}
|
`}
|
||||||
<div>
|
<div>
|
||||||
<div class="title">
|
<div class="title">${this.title}</div>
|
||||||
${this.title}
|
|
||||||
</div>
|
|
||||||
<div class="addition">
|
<div class="addition">
|
||||||
${this.description}
|
${this.description}
|
||||||
${/* treat as available when undefined */
|
${
|
||||||
this.available === false ? " (Not available)" : ""}
|
/* treat as available when undefined */
|
||||||
|
this.available === false ? " (Not available)" : ""
|
||||||
|
}
|
||||||
${this.datetime
|
${this.datetime
|
||||||
? html`
|
? html`
|
||||||
<ha-relative-time
|
<ha-relative-time
|
||||||
@@ -77,7 +70,7 @@ class HassioCardContent extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
ha-svg-icon {
|
ha-svg-icon {
|
||||||
margin-right: 24px;
|
margin-right: 24px;
|
||||||
|
@@ -2,13 +2,8 @@ import "@material/mwc-icon-button/mwc-icon-button";
|
|||||||
import { mdiFolderUpload } from "@mdi/js";
|
import { mdiFolderUpload } from "@mdi/js";
|
||||||
import "@polymer/iron-input/iron-input";
|
import "@polymer/iron-input/iron-input";
|
||||||
import "@polymer/paper-input/paper-input-container";
|
import "@polymer/paper-input/paper-input-container";
|
||||||
import {
|
import { html, LitElement, TemplateResult } from "lit";
|
||||||
customElement,
|
import { customElement, state } from "lit/decorators";
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { fireEvent } from "../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../src/common/dom/fire_event";
|
||||||
import "../../../src/components/ha-circular-progress";
|
import "../../../src/components/ha-circular-progress";
|
||||||
import "../../../src/components/ha-file-upload";
|
import "../../../src/components/ha-file-upload";
|
||||||
@@ -33,9 +28,9 @@ const MAX_FILE_SIZE = 1 * 1024 * 1024 * 1024; // 1GB
|
|||||||
export class HassioUploadSnapshot extends LitElement {
|
export class HassioUploadSnapshot extends LitElement {
|
||||||
public hass!: HomeAssistant;
|
public hass!: HomeAssistant;
|
||||||
|
|
||||||
@internalProperty() public value: string | null = null;
|
@state() public value: string | null = null;
|
||||||
|
|
||||||
@internalProperty() private _uploading = false;
|
@state() private _uploading = false;
|
||||||
|
|
||||||
public render(): TemplateResult {
|
public render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
|
54
hassio/src/components/supervisor-formfield-label.ts
Normal file
54
hassio/src/components/supervisor-formfield-label.ts
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement, property } from "lit/decorators";
|
||||||
|
import "../../../src/components/ha-svg-icon";
|
||||||
|
|
||||||
|
@customElement("supervisor-formfield-label")
|
||||||
|
class SupervisorFormfieldLabel extends LitElement {
|
||||||
|
@property({ type: String }) public label!: string;
|
||||||
|
|
||||||
|
@property({ type: String }) public imageUrl?: string;
|
||||||
|
|
||||||
|
@property({ type: String }) public iconPath?: string;
|
||||||
|
|
||||||
|
@property({ type: String }) public version?: string;
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
${this.imageUrl
|
||||||
|
? html`<img loading="lazy" .src=${this.imageUrl} class="icon" />`
|
||||||
|
: this.iconPath
|
||||||
|
? html`<ha-svg-icon .path=${this.iconPath} class="icon"></ha-svg-icon>`
|
||||||
|
: ""}
|
||||||
|
<span class="label">${this.label}</span>
|
||||||
|
${this.version
|
||||||
|
? html`<span class="version">(${this.version})</span>`
|
||||||
|
: ""}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResultGroup {
|
||||||
|
return css`
|
||||||
|
:host {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
.label {
|
||||||
|
margin-right: 4px;
|
||||||
|
}
|
||||||
|
.version {
|
||||||
|
color: var(--secondary-text-color);
|
||||||
|
}
|
||||||
|
.icon {
|
||||||
|
max-height: 22px;
|
||||||
|
max-width: 22px;
|
||||||
|
margin-right: 8px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"supervisor-formfield-label": SupervisorFormfieldLabel;
|
||||||
|
}
|
||||||
|
}
|
@@ -1,13 +1,6 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
import { classMap } from "lit/directives/class-map";
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { classMap } from "lit-html/directives/class-map";
|
|
||||||
import "../../../src/components/ha-bar";
|
import "../../../src/components/ha-bar";
|
||||||
import "../../../src/components/ha-settings-row";
|
import "../../../src/components/ha-settings-row";
|
||||||
import { roundWithOneDecimal } from "../../../src/util/calculate";
|
import { roundWithOneDecimal } from "../../../src/util/calculate";
|
||||||
@@ -23,13 +16,9 @@ class SupervisorMetric extends LitElement {
|
|||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
const roundedValue = roundWithOneDecimal(this.value);
|
const roundedValue = roundWithOneDecimal(this.value);
|
||||||
return html`<ha-settings-row>
|
return html`<ha-settings-row>
|
||||||
<span slot="heading">
|
<span slot="heading"> ${this.description} </span>
|
||||||
${this.description}
|
<div slot="description" .title=${this.tooltip ?? ""}>
|
||||||
</span>
|
<span class="value"> ${roundedValue} % </span>
|
||||||
<div slot="description" title="${this.tooltip ?? ""}">
|
|
||||||
<span class="value">
|
|
||||||
${roundedValue}%
|
|
||||||
</span>
|
|
||||||
<ha-bar
|
<ha-bar
|
||||||
class="${classMap({
|
class="${classMap({
|
||||||
"target-warning": roundedValue > 50,
|
"target-warning": roundedValue > 50,
|
||||||
@@ -41,7 +30,7 @@ class SupervisorMetric extends LitElement {
|
|||||||
</ha-settings-row>`;
|
</ha-settings-row>`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult {
|
static get styles(): CSSResultGroup {
|
||||||
return css`
|
return css`
|
||||||
ha-settings-row {
|
ha-settings-row {
|
||||||
padding: 0;
|
padding: 0;
|
||||||
@@ -73,8 +62,9 @@ class SupervisorMetric extends LitElement {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
.value {
|
.value {
|
||||||
width: 42px;
|
width: 48px;
|
||||||
padding-right: 4px;
|
padding-right: 4px;
|
||||||
|
flex-shrink: 0;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
450
hassio/src/components/supervisor-snapshot-content.ts
Normal file
450
hassio/src/components/supervisor-snapshot-content.ts
Normal file
@@ -0,0 +1,450 @@
|
|||||||
|
import { mdiFolder, mdiHomeAssistant, mdiPuzzle } from "@mdi/js";
|
||||||
|
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
||||||
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement, property } from "lit/decorators";
|
||||||
|
import { atLeastVersion } from "../../../src/common/config/version";
|
||||||
|
import { formatDate } from "../../../src/common/datetime/format_date";
|
||||||
|
import { formatDateTime } from "../../../src/common/datetime/format_date_time";
|
||||||
|
import { LocalizeFunc } from "../../../src/common/translations/localize";
|
||||||
|
import "../../../src/components/ha-checkbox";
|
||||||
|
import "../../../src/components/ha-formfield";
|
||||||
|
import "../../../src/components/ha-radio";
|
||||||
|
import type { HaRadio } from "../../../src/components/ha-radio";
|
||||||
|
import {
|
||||||
|
HassioFullSnapshotCreateParams,
|
||||||
|
HassioPartialSnapshotCreateParams,
|
||||||
|
HassioSnapshotDetail,
|
||||||
|
} from "../../../src/data/hassio/snapshot";
|
||||||
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
|
import { PolymerChangedEvent } from "../../../src/polymer-types";
|
||||||
|
import { HomeAssistant } from "../../../src/types";
|
||||||
|
import "./supervisor-formfield-label";
|
||||||
|
|
||||||
|
interface CheckboxItem {
|
||||||
|
slug: string;
|
||||||
|
checked: boolean;
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface AddonCheckboxItem extends CheckboxItem {
|
||||||
|
version: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const _computeFolders = (folders): CheckboxItem[] => {
|
||||||
|
const list: CheckboxItem[] = [];
|
||||||
|
if (folders.includes("homeassistant")) {
|
||||||
|
list.push({
|
||||||
|
slug: "homeassistant",
|
||||||
|
name: "Home Assistant configuration",
|
||||||
|
checked: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (folders.includes("ssl")) {
|
||||||
|
list.push({ slug: "ssl", name: "SSL", checked: false });
|
||||||
|
}
|
||||||
|
if (folders.includes("share")) {
|
||||||
|
list.push({ slug: "share", name: "Share", checked: false });
|
||||||
|
}
|
||||||
|
if (folders.includes("media")) {
|
||||||
|
list.push({ slug: "media", name: "Media", checked: false });
|
||||||
|
}
|
||||||
|
if (folders.includes("addons/local")) {
|
||||||
|
list.push({ slug: "addons/local", name: "Local add-ons", checked: false });
|
||||||
|
}
|
||||||
|
return list.sort((a, b) => (a.name > b.name ? 1 : -1));
|
||||||
|
};
|
||||||
|
|
||||||
|
const _computeAddons = (addons): AddonCheckboxItem[] =>
|
||||||
|
addons
|
||||||
|
.map((addon) => ({
|
||||||
|
slug: addon.slug,
|
||||||
|
name: addon.name,
|
||||||
|
version: addon.version,
|
||||||
|
checked: false,
|
||||||
|
}))
|
||||||
|
.sort((a, b) => (a.name > b.name ? 1 : -1));
|
||||||
|
|
||||||
|
@customElement("supervisor-snapshot-content")
|
||||||
|
export class SupervisorSnapshotContent extends LitElement {
|
||||||
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property() public localize?: LocalizeFunc;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor?: Supervisor;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public snapshot?: HassioSnapshotDetail;
|
||||||
|
|
||||||
|
@property() public snapshotType: HassioSnapshotDetail["type"] = "full";
|
||||||
|
|
||||||
|
@property({ attribute: false }) public folders?: CheckboxItem[];
|
||||||
|
|
||||||
|
@property({ attribute: false }) public addons?: AddonCheckboxItem[];
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public homeAssistant = false;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public snapshotHasPassword = false;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public onboarding = false;
|
||||||
|
|
||||||
|
@property() public snapshotName = "";
|
||||||
|
|
||||||
|
@property() public snapshotPassword = "";
|
||||||
|
|
||||||
|
@property() public confirmSnapshotPassword = "";
|
||||||
|
|
||||||
|
public willUpdate(changedProps) {
|
||||||
|
super.willUpdate(changedProps);
|
||||||
|
if (!this.hasUpdated) {
|
||||||
|
this.folders = _computeFolders(
|
||||||
|
this.snapshot
|
||||||
|
? this.snapshot.folders
|
||||||
|
: ["homeassistant", "ssl", "share", "media", "addons/local"]
|
||||||
|
);
|
||||||
|
this.addons = _computeAddons(
|
||||||
|
this.snapshot
|
||||||
|
? this.snapshot.addons
|
||||||
|
: this.supervisor?.supervisor.addons
|
||||||
|
);
|
||||||
|
this.snapshotType = this.snapshot?.type || "full";
|
||||||
|
this.snapshotName = this.snapshot?.name || "";
|
||||||
|
this.snapshotHasPassword = this.snapshot?.protected || false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private _localize = (string: string) =>
|
||||||
|
this.supervisor?.localize(`snapshot.${string}`) ||
|
||||||
|
this.localize!(`ui.panel.page-onboarding.restore.${string}`);
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this.onboarding && !this.supervisor) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
const foldersSection =
|
||||||
|
this.snapshotType === "partial" ? this._getSection("folders") : undefined;
|
||||||
|
const addonsSection =
|
||||||
|
this.snapshotType === "partial" ? this._getSection("addons") : undefined;
|
||||||
|
|
||||||
|
return html`
|
||||||
|
${this.snapshot
|
||||||
|
? html`<div class="details">
|
||||||
|
${this.snapshot.type === "full"
|
||||||
|
? this._localize("full_snapshot")
|
||||||
|
: this._localize("partial_snapshot")}
|
||||||
|
(${Math.ceil(this.snapshot.size * 10) / 10 + " MB"})<br />
|
||||||
|
${this.hass
|
||||||
|
? formatDateTime(new Date(this.snapshot.date), this.hass.locale)
|
||||||
|
: this.snapshot.date}
|
||||||
|
</div>`
|
||||||
|
: html`<paper-input
|
||||||
|
name="snapshotName"
|
||||||
|
.label=${this.supervisor?.localize("snapshot.name") || "Name"}
|
||||||
|
.value=${this.snapshotName}
|
||||||
|
@value-changed=${this._handleTextValueChanged}
|
||||||
|
>
|
||||||
|
</paper-input>`}
|
||||||
|
${!this.snapshot || this.snapshot.type === "full"
|
||||||
|
? html`<div class="sub-header">
|
||||||
|
${!this.snapshot
|
||||||
|
? this._localize("type")
|
||||||
|
: this._localize("select_type")}
|
||||||
|
</div>
|
||||||
|
<div class="snapshot-types">
|
||||||
|
<ha-formfield .label=${this._localize("full_snapshot")}>
|
||||||
|
<ha-radio
|
||||||
|
@change=${this._handleRadioValueChanged}
|
||||||
|
value="full"
|
||||||
|
name="snapshotType"
|
||||||
|
.checked=${this.snapshotType === "full"}
|
||||||
|
>
|
||||||
|
</ha-radio>
|
||||||
|
</ha-formfield>
|
||||||
|
<ha-formfield .label=${this._localize("partial_snapshot")}>
|
||||||
|
<ha-radio
|
||||||
|
@change=${this._handleRadioValueChanged}
|
||||||
|
value="partial"
|
||||||
|
name="snapshotType"
|
||||||
|
.checked=${this.snapshotType === "partial"}
|
||||||
|
>
|
||||||
|
</ha-radio>
|
||||||
|
</ha-formfield>
|
||||||
|
</div>`
|
||||||
|
: ""}
|
||||||
|
${this.snapshotType === "partial"
|
||||||
|
? html`<div class="partial-picker">
|
||||||
|
${this.snapshot && this.snapshot.homeassistant
|
||||||
|
? html`
|
||||||
|
<ha-formfield
|
||||||
|
.label=${html`<supervisor-formfield-label
|
||||||
|
label="Home Assistant"
|
||||||
|
.iconPath=${mdiHomeAssistant}
|
||||||
|
.version=${this.snapshot.homeassistant}
|
||||||
|
>
|
||||||
|
</supervisor-formfield-label>`}
|
||||||
|
>
|
||||||
|
<ha-checkbox
|
||||||
|
.checked=${this.homeAssistant}
|
||||||
|
@click=${() => {
|
||||||
|
this.homeAssistant = !this.homeAssistant;
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
</ha-checkbox>
|
||||||
|
</ha-formfield>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
${foldersSection?.templates.length
|
||||||
|
? html`
|
||||||
|
<ha-formfield
|
||||||
|
.label=${html`<supervisor-formfield-label
|
||||||
|
.label=${this._localize("folders")}
|
||||||
|
.iconPath=${mdiFolder}
|
||||||
|
>
|
||||||
|
</supervisor-formfield-label>`}
|
||||||
|
>
|
||||||
|
<ha-checkbox
|
||||||
|
@change=${this._toggleSection}
|
||||||
|
.checked=${foldersSection.checked}
|
||||||
|
.indeterminate=${foldersSection.indeterminate}
|
||||||
|
.section=${"folders"}
|
||||||
|
>
|
||||||
|
</ha-checkbox>
|
||||||
|
</ha-formfield>
|
||||||
|
<div class="section-content">${foldersSection.templates}</div>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
${addonsSection?.templates.length
|
||||||
|
? html`
|
||||||
|
<ha-formfield
|
||||||
|
.label=${html`<supervisor-formfield-label
|
||||||
|
.label=${this._localize("addons")}
|
||||||
|
.iconPath=${mdiPuzzle}
|
||||||
|
>
|
||||||
|
</supervisor-formfield-label>`}
|
||||||
|
>
|
||||||
|
<ha-checkbox
|
||||||
|
@change=${this._toggleSection}
|
||||||
|
.checked=${addonsSection.checked}
|
||||||
|
.indeterminate=${addonsSection.indeterminate}
|
||||||
|
.section=${"addons"}
|
||||||
|
>
|
||||||
|
</ha-checkbox>
|
||||||
|
</ha-formfield>
|
||||||
|
<div class="section-content">${addonsSection.templates}</div>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
</div> `
|
||||||
|
: ""}
|
||||||
|
${this.snapshotType === "partial" &&
|
||||||
|
(!this.snapshot || this.snapshotHasPassword)
|
||||||
|
? html`<hr />`
|
||||||
|
: ""}
|
||||||
|
${!this.snapshot
|
||||||
|
? html`<ha-formfield
|
||||||
|
class="password"
|
||||||
|
.label=${this._localize("password_protection")}
|
||||||
|
>
|
||||||
|
<ha-checkbox
|
||||||
|
.checked=${this.snapshotHasPassword}
|
||||||
|
@change=${this._toggleHasPassword}
|
||||||
|
>
|
||||||
|
</ha-checkbox>
|
||||||
|
</ha-formfield>`
|
||||||
|
: ""}
|
||||||
|
${this.snapshotHasPassword
|
||||||
|
? html`
|
||||||
|
<paper-input
|
||||||
|
.label=${this._localize("password")}
|
||||||
|
type="password"
|
||||||
|
name="snapshotPassword"
|
||||||
|
.value=${this.snapshotPassword}
|
||||||
|
@value-changed=${this._handleTextValueChanged}
|
||||||
|
>
|
||||||
|
</paper-input>
|
||||||
|
${!this.snapshot
|
||||||
|
? html` <paper-input
|
||||||
|
.label=${this.supervisor?.localize("confirm_password")}
|
||||||
|
type="password"
|
||||||
|
name="confirmSnapshotPassword"
|
||||||
|
.value=${this.confirmSnapshotPassword}
|
||||||
|
@value-changed=${this._handleTextValueChanged}
|
||||||
|
>
|
||||||
|
</paper-input>`
|
||||||
|
: ""}
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResultGroup {
|
||||||
|
return css`
|
||||||
|
.partial-picker ha-formfield {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.partial-picker ha-checkbox {
|
||||||
|
--mdc-checkbox-touch-target-size: 32px;
|
||||||
|
}
|
||||||
|
.partial-picker {
|
||||||
|
display: block;
|
||||||
|
margin: 0px -6px;
|
||||||
|
}
|
||||||
|
supervisor-formfield-label {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
hr {
|
||||||
|
border-color: var(--divider-color);
|
||||||
|
border-bottom: none;
|
||||||
|
margin: 16px 0;
|
||||||
|
}
|
||||||
|
.details {
|
||||||
|
color: var(--secondary-text-color);
|
||||||
|
}
|
||||||
|
.section-content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
margin-left: 30px;
|
||||||
|
}
|
||||||
|
ha-formfield.password {
|
||||||
|
display: block;
|
||||||
|
margin: 0 -14px -16px;
|
||||||
|
}
|
||||||
|
.snapshot-types {
|
||||||
|
display: flex;
|
||||||
|
margin-left: -13px;
|
||||||
|
}
|
||||||
|
.sub-header {
|
||||||
|
margin-top: 8px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
public snapshotDetails():
|
||||||
|
| HassioPartialSnapshotCreateParams
|
||||||
|
| HassioFullSnapshotCreateParams {
|
||||||
|
const data: any = {};
|
||||||
|
|
||||||
|
if (!this.snapshot) {
|
||||||
|
data.name = this.snapshotName || formatDate(new Date(), this.hass.locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.snapshotHasPassword) {
|
||||||
|
data.password = this.snapshotPassword;
|
||||||
|
if (!this.snapshot) {
|
||||||
|
data.confirm_password = this.confirmSnapshotPassword;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.snapshotType === "full") {
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
const addons = this.addons
|
||||||
|
?.filter((addon) => addon.checked)
|
||||||
|
.map((addon) => addon.slug);
|
||||||
|
const folders = this.folders
|
||||||
|
?.filter((folder) => folder.checked)
|
||||||
|
.map((folder) => folder.slug);
|
||||||
|
|
||||||
|
if (addons?.length) {
|
||||||
|
data.addons = addons;
|
||||||
|
}
|
||||||
|
if (folders?.length) {
|
||||||
|
data.folders = folders;
|
||||||
|
}
|
||||||
|
if (this.homeAssistant) {
|
||||||
|
data.homeassistant = this.homeAssistant;
|
||||||
|
}
|
||||||
|
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _getSection(section: string) {
|
||||||
|
const templates: TemplateResult[] = [];
|
||||||
|
const addons =
|
||||||
|
section === "addons"
|
||||||
|
? new Map(
|
||||||
|
this.supervisor?.addon.addons.map((item) => [item.slug, item])
|
||||||
|
)
|
||||||
|
: undefined;
|
||||||
|
let checkedItems = 0;
|
||||||
|
this[section].forEach((item) => {
|
||||||
|
templates.push(html`<ha-formfield
|
||||||
|
.label=${html`<supervisor-formfield-label
|
||||||
|
.label=${item.name}
|
||||||
|
.iconPath=${section === "addons" ? mdiPuzzle : mdiFolder}
|
||||||
|
.imageUrl=${section === "addons" &&
|
||||||
|
!this.onboarding &&
|
||||||
|
atLeastVersion(this.hass.config.version, 0, 105) &&
|
||||||
|
addons?.get(item.slug)?.icon
|
||||||
|
? `/api/hassio/addons/${item.slug}/icon`
|
||||||
|
: undefined}
|
||||||
|
.version=${item.version}
|
||||||
|
>
|
||||||
|
</supervisor-formfield-label>`}
|
||||||
|
>
|
||||||
|
<ha-checkbox
|
||||||
|
.item=${item}
|
||||||
|
.checked=${item.checked}
|
||||||
|
.section=${section}
|
||||||
|
@change=${this._updateSectionEntry}
|
||||||
|
>
|
||||||
|
</ha-checkbox>
|
||||||
|
</ha-formfield>`);
|
||||||
|
|
||||||
|
if (item.checked) {
|
||||||
|
checkedItems++;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const checked = checkedItems === this[section].length;
|
||||||
|
|
||||||
|
return {
|
||||||
|
templates,
|
||||||
|
checked,
|
||||||
|
indeterminate: !checked && checkedItems !== 0,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleRadioValueChanged(ev: CustomEvent) {
|
||||||
|
const input = ev.currentTarget as HaRadio;
|
||||||
|
this[input.name] = input.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleTextValueChanged(ev: PolymerChangedEvent<string>) {
|
||||||
|
const input = ev.currentTarget as PaperInputElement;
|
||||||
|
this[input.name!] = ev.detail.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _toggleHasPassword(): void {
|
||||||
|
this.snapshotHasPassword = !this.snapshotHasPassword;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _toggleSection(ev): void {
|
||||||
|
const section = ev.currentTarget.section;
|
||||||
|
|
||||||
|
this[section] = (section === "addons" ? this.addons : this.folders)!.map(
|
||||||
|
(item) => ({
|
||||||
|
...item,
|
||||||
|
checked: ev.currentTarget.checked,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _updateSectionEntry(ev): void {
|
||||||
|
const item = ev.currentTarget.item;
|
||||||
|
const section = ev.currentTarget.section;
|
||||||
|
this[section] = this[section].map((entry) =>
|
||||||
|
entry.slug === item.slug
|
||||||
|
? {
|
||||||
|
...entry,
|
||||||
|
checked: ev.currentTarget.checked,
|
||||||
|
}
|
||||||
|
: entry
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"supervisor-snapshot-content": SupervisorSnapshotContent;
|
||||||
|
}
|
||||||
|
}
|
@@ -1,13 +1,6 @@
|
|||||||
import { mdiArrowUpBoldCircle, mdiPuzzle } from "@mdi/js";
|
import { mdiArrowUpBoldCircle, mdiPuzzle } from "@mdi/js";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { atLeastVersion } from "../../../src/common/config/version";
|
import { atLeastVersion } from "../../../src/common/config/version";
|
||||||
import { navigate } from "../../../src/common/navigate";
|
import { navigate } from "../../../src/common/navigate";
|
||||||
import { compare } from "../../../src/common/string/compare";
|
import { compare } from "../../../src/common/string/compare";
|
||||||
@@ -27,17 +20,15 @@ class HassioAddons extends LitElement {
|
|||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<h1>Add-ons</h1>
|
<h1>${this.supervisor.localize("dashboard.addons")}</h1>
|
||||||
<div class="card-group">
|
<div class="card-group">
|
||||||
${!this.supervisor.supervisor.addons?.length
|
${!this.supervisor.supervisor.addons?.length
|
||||||
? html`
|
? html`
|
||||||
<ha-card>
|
<ha-card>
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
You don't have any add-ons installed yet. Head over to
|
|
||||||
<button class="link" @click=${this._openStore}>
|
<button class="link" @click=${this._openStore}>
|
||||||
the add-on store
|
${this.supervisor.localize("dashboard.no_addons")}
|
||||||
</button>
|
</button>
|
||||||
to get started!
|
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
`
|
`
|
||||||
@@ -58,10 +49,16 @@ class HassioAddons extends LitElement {
|
|||||||
? mdiArrowUpBoldCircle
|
? mdiArrowUpBoldCircle
|
||||||
: mdiPuzzle}
|
: mdiPuzzle}
|
||||||
.iconTitle=${addon.state !== "started"
|
.iconTitle=${addon.state !== "started"
|
||||||
? "Add-on is stopped"
|
? this.supervisor.localize(
|
||||||
|
"dashboard.addon_stopped"
|
||||||
|
)
|
||||||
: addon.update_available!
|
: addon.update_available!
|
||||||
? "New version available"
|
? this.supervisor.localize(
|
||||||
: "Add-on is running"}
|
"dashboard.addon_new_version"
|
||||||
|
)
|
||||||
|
: this.supervisor.localize(
|
||||||
|
"dashboard.addon_running"
|
||||||
|
)}
|
||||||
.iconClass=${addon.update_available
|
.iconClass=${addon.update_available
|
||||||
? addon.state === "started"
|
? addon.state === "started"
|
||||||
? "update"
|
? "update"
|
||||||
@@ -86,7 +83,7 @@ class HassioAddons extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
@@ -99,11 +96,11 @@ class HassioAddons extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private _addonTapped(ev: any): void {
|
private _addonTapped(ev: any): void {
|
||||||
navigate(this, `/hassio/addon/${ev.currentTarget.addon.slug}/info`);
|
navigate(`/hassio/addon/${ev.currentTarget.addon.slug}/info`);
|
||||||
}
|
}
|
||||||
|
|
||||||
private _openStore(): void {
|
private _openStore(): void {
|
||||||
navigate(this, "/hassio/store");
|
navigate("/hassio/store");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,12 +1,5 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import "../../../src/layouts/hass-tabs-subpage";
|
import "../../../src/layouts/hass-tabs-subpage";
|
||||||
import { haStyle } from "../../../src/resources/styles";
|
import { haStyle } from "../../../src/resources/styles";
|
||||||
@@ -29,13 +22,16 @@ class HassioDashboard extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
<hass-tabs-subpage
|
<hass-tabs-subpage
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.localizeFunc=${this.supervisor.localize}
|
||||||
.narrow=${this.narrow}
|
.narrow=${this.narrow}
|
||||||
hassio
|
|
||||||
main-page
|
|
||||||
.route=${this.route}
|
.route=${this.route}
|
||||||
.tabs=${supervisorTabs}
|
.tabs=${supervisorTabs}
|
||||||
|
main-page
|
||||||
|
supervisor
|
||||||
>
|
>
|
||||||
<span slot="header">Dashboard</span>
|
<span slot="header">
|
||||||
|
${this.supervisor.localize("panel.dashboard")}
|
||||||
|
</span>
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<hassio-update
|
<hassio-update
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
@@ -50,7 +46,7 @@ class HassioDashboard extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
css`
|
css`
|
||||||
|
@@ -1,50 +1,53 @@
|
|||||||
import "@material/mwc-button";
|
import "@material/mwc-button";
|
||||||
import { mdiHomeAssistant } from "@mdi/js";
|
import { mdiHomeAssistant } from "@mdi/js";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
|
import { atLeastVersion } from "../../../src/common/config/version";
|
||||||
import { fireEvent } from "../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../src/common/dom/fire_event";
|
||||||
import "../../../src/components/buttons/ha-progress-button";
|
import "../../../src/components/buttons/ha-progress-button";
|
||||||
import "../../../src/components/ha-card";
|
import "../../../src/components/ha-card";
|
||||||
|
import "../../../src/components/ha-settings-row";
|
||||||
import "../../../src/components/ha-svg-icon";
|
import "../../../src/components/ha-svg-icon";
|
||||||
import {
|
import {
|
||||||
extractApiErrorMessage,
|
extractApiErrorMessage,
|
||||||
HassioResponse,
|
HassioResponse,
|
||||||
ignoredStatusCodes,
|
ignoreSupervisorError,
|
||||||
} from "../../../src/data/hassio/common";
|
} from "../../../src/data/hassio/common";
|
||||||
import { HassioHassOSInfo } from "../../../src/data/hassio/host";
|
import { HassioHassOSInfo } from "../../../src/data/hassio/host";
|
||||||
import {
|
import {
|
||||||
HassioHomeAssistantInfo,
|
HassioHomeAssistantInfo,
|
||||||
HassioSupervisorInfo,
|
HassioSupervisorInfo,
|
||||||
} from "../../../src/data/hassio/supervisor";
|
} from "../../../src/data/hassio/supervisor";
|
||||||
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
import { updateCore } from "../../../src/data/supervisor/core";
|
||||||
|
import {
|
||||||
|
Supervisor,
|
||||||
|
supervisorApiWsRequest,
|
||||||
|
} from "../../../src/data/supervisor/supervisor";
|
||||||
import {
|
import {
|
||||||
showAlertDialog,
|
showAlertDialog,
|
||||||
showConfirmationDialog,
|
showConfirmationDialog,
|
||||||
} from "../../../src/dialogs/generic/show-dialog-box";
|
} from "../../../src/dialogs/generic/show-dialog-box";
|
||||||
import { haStyle } from "../../../src/resources/styles";
|
import { haStyle } from "../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../src/types";
|
import { HomeAssistant } from "../../../src/types";
|
||||||
import { showDialogSupervisorCoreUpdate } from "../dialogs/core/show-dialog-core-update";
|
import { showDialogSupervisorUpdate } from "../dialogs/update/show-dialog-update";
|
||||||
import { hassioStyle } from "../resources/hassio-style";
|
import { hassioStyle } from "../resources/hassio-style";
|
||||||
|
|
||||||
|
const computeVersion = (key: string, version: string): string =>
|
||||||
|
key === "os" ? version : `${key}-${version}`;
|
||||||
|
|
||||||
@customElement("hassio-update")
|
@customElement("hassio-update")
|
||||||
export class HassioUpdate extends LitElement {
|
export class HassioUpdate extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property({ attribute: false }) public supervisor!: Supervisor;
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
private _pendingUpdates = memoizeOne((supervisor: Supervisor): number => {
|
private _pendingUpdates = memoizeOne(
|
||||||
return Object.keys(supervisor).filter(
|
(supervisor: Supervisor): number =>
|
||||||
(value) => supervisor[value].update_available
|
Object.keys(supervisor).filter(
|
||||||
).length;
|
(value) => supervisor[value].update_available
|
||||||
});
|
).length
|
||||||
|
);
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
if (!this.supervisor) {
|
if (!this.supervisor) {
|
||||||
@@ -59,9 +62,12 @@ export class HassioUpdate extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<h1>
|
<h1>
|
||||||
${updatesAvailable > 1
|
${this.supervisor.localize(
|
||||||
? "Updates Available 🎉"
|
"common.update_available",
|
||||||
: "Update Available 🎉"}
|
"count",
|
||||||
|
updatesAvailable
|
||||||
|
)}
|
||||||
|
🎉
|
||||||
</h1>
|
</h1>
|
||||||
<div class="card-group">
|
<div class="card-group">
|
||||||
${this._renderUpdateCard(
|
${this._renderUpdateCard(
|
||||||
@@ -110,14 +116,30 @@ export class HassioUpdate extends LitElement {
|
|||||||
<div class="icon">
|
<div class="icon">
|
||||||
<ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiHomeAssistant}></ha-svg-icon>
|
||||||
</div>
|
</div>
|
||||||
<div class="update-heading">${name} ${object.version_latest}</div>
|
<div class="update-heading">${name}</div>
|
||||||
<div class="warning">
|
<ha-settings-row two-line>
|
||||||
You are currently running version ${object.version}
|
<span slot="heading">
|
||||||
</div>
|
${this.supervisor.localize("common.version")}
|
||||||
|
</span>
|
||||||
|
<span slot="description">
|
||||||
|
${computeVersion(key, object.version!)}
|
||||||
|
</span>
|
||||||
|
</ha-settings-row>
|
||||||
|
|
||||||
|
<ha-settings-row two-line>
|
||||||
|
<span slot="heading">
|
||||||
|
${this.supervisor.localize("common.newest_version")}
|
||||||
|
</span>
|
||||||
|
<span slot="description">
|
||||||
|
${computeVersion(key, object.version_latest!)}
|
||||||
|
</span>
|
||||||
|
</ha-settings-row>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
<a href="${releaseNotesUrl}" target="_blank" rel="noreferrer">
|
<a href="${releaseNotesUrl}" target="_blank" rel="noreferrer">
|
||||||
<mwc-button>Release notes</mwc-button>
|
<mwc-button>
|
||||||
|
${this.supervisor.localize("common.release_notes")}
|
||||||
|
</mwc-button>
|
||||||
</a>
|
</a>
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
.apiPath=${apiPath}
|
.apiPath=${apiPath}
|
||||||
@@ -126,7 +148,7 @@ export class HassioUpdate extends LitElement {
|
|||||||
.version=${object.version_latest}
|
.version=${object.version_latest}
|
||||||
@click=${this._confirmUpdate}
|
@click=${this._confirmUpdate}
|
||||||
>
|
>
|
||||||
Update
|
${this.supervisor.localize("common.update")}
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
@@ -136,15 +158,35 @@ export class HassioUpdate extends LitElement {
|
|||||||
private async _confirmUpdate(ev): Promise<void> {
|
private async _confirmUpdate(ev): Promise<void> {
|
||||||
const item = ev.currentTarget;
|
const item = ev.currentTarget;
|
||||||
if (item.key === "core") {
|
if (item.key === "core") {
|
||||||
showDialogSupervisorCoreUpdate(this, { core: this.supervisor.core });
|
showDialogSupervisorUpdate(this, {
|
||||||
|
supervisor: this.supervisor,
|
||||||
|
name: "Home Assistant Core",
|
||||||
|
version: this.supervisor.core.version_latest,
|
||||||
|
snapshotParams: {
|
||||||
|
name: `core_${this.supervisor.core.version}`,
|
||||||
|
folders: ["homeassistant"],
|
||||||
|
homeassistant: true,
|
||||||
|
},
|
||||||
|
updateHandler: async () => this._updateCore(),
|
||||||
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
item.progress = true;
|
item.progress = true;
|
||||||
const confirmed = await showConfirmationDialog(this, {
|
const confirmed = await showConfirmationDialog(this, {
|
||||||
title: `Update ${item.name}`,
|
title: this.supervisor.localize(
|
||||||
text: `Are you sure you want to update ${item.name} to version ${item.version}?`,
|
"confirm.update.title",
|
||||||
confirmText: "update",
|
"name",
|
||||||
dismissText: "cancel",
|
item.name
|
||||||
|
),
|
||||||
|
text: this.supervisor.localize(
|
||||||
|
"confirm.update.text",
|
||||||
|
"name",
|
||||||
|
item.name,
|
||||||
|
"version",
|
||||||
|
computeVersion(item.key, item.version)
|
||||||
|
),
|
||||||
|
confirmText: this.supervisor.localize("common.update"),
|
||||||
|
dismissText: this.supervisor.localize("common.cancel"),
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!confirmed) {
|
if (!confirmed) {
|
||||||
@@ -152,20 +194,24 @@ export class HassioUpdate extends LitElement {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
await this.hass.callApi<HassioResponse<void>>("POST", item.apiPath);
|
if (atLeastVersion(this.hass.config.version, 2021, 2, 4)) {
|
||||||
fireEvent(this, "supervisor-colllection-refresh", {
|
await supervisorApiWsRequest(this.hass.connection, {
|
||||||
colllection: item.key,
|
method: "post",
|
||||||
|
endpoint: item.apiPath.replace("hassio", ""),
|
||||||
|
timeout: null,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
await this.hass.callApi<HassioResponse<void>>("POST", item.apiPath);
|
||||||
|
}
|
||||||
|
fireEvent(this, "supervisor-collection-refresh", {
|
||||||
|
collection: item.key,
|
||||||
});
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
// Only show an error if the status code was not expected (user behind proxy)
|
// Only show an error if the status code was not expected (user behind proxy)
|
||||||
// or no status at all(connection terminated)
|
// or no status at all(connection terminated)
|
||||||
if (
|
if (this.hass.connection.connected && !ignoreSupervisorError(err)) {
|
||||||
this.hass.connection.connected &&
|
|
||||||
err.status_code &&
|
|
||||||
!ignoredStatusCodes.has(err.status_code)
|
|
||||||
) {
|
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Update failed",
|
title: this.supervisor.localize("common.error.update_failed"),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -173,7 +219,33 @@ export class HassioUpdate extends LitElement {
|
|||||||
item.progress = false;
|
item.progress = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
private async _updateCore(): Promise<void> {
|
||||||
|
try {
|
||||||
|
await updateCore(this.hass);
|
||||||
|
} catch (err) {
|
||||||
|
if (this.hass.connection.connected && !ignoreSupervisorError(err)) {
|
||||||
|
showAlertDialog(this, {
|
||||||
|
title: this.supervisor.localize(
|
||||||
|
"common.failed_to_update_name",
|
||||||
|
"name",
|
||||||
|
"Home Assistant Core"
|
||||||
|
),
|
||||||
|
text: extractApiErrorMessage(err),
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fireEvent(this, "supervisor-collection-refresh", {
|
||||||
|
collection: "core",
|
||||||
|
});
|
||||||
|
fireEvent(this, "supervisor-applying-update", {
|
||||||
|
name: "Home Assistant Core",
|
||||||
|
version: this.supervisor.core.version_latest,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
@@ -190,9 +262,6 @@ export class HassioUpdate extends LitElement {
|
|||||||
margin-bottom: 0.5em;
|
margin-bottom: 0.5em;
|
||||||
color: var(--primary-text-color);
|
color: var(--primary-text-color);
|
||||||
}
|
}
|
||||||
.warning {
|
|
||||||
color: var(--secondary-text-color);
|
|
||||||
}
|
|
||||||
.card-content {
|
.card-content {
|
||||||
height: calc(100% - 47px);
|
height: calc(100% - 47px);
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
@@ -200,13 +269,13 @@ export class HassioUpdate extends LitElement {
|
|||||||
.card-actions {
|
.card-actions {
|
||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
.errors {
|
|
||||||
color: var(--error-color);
|
|
||||||
padding: 16px;
|
|
||||||
}
|
|
||||||
a {
|
a {
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
}
|
}
|
||||||
|
ha-settings-row {
|
||||||
|
padding: 0;
|
||||||
|
--paper-item-body-two-line-min-height: 32px;
|
||||||
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
@@ -1,17 +0,0 @@
|
|||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
|
||||||
import { HassioAddonDetails } from "../../../../src/data/hassio/addon";
|
|
||||||
|
|
||||||
export interface SupervisorDialogSupervisorAddonUpdateParams {
|
|
||||||
addon: HassioAddonDetails;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const showDialogSupervisorAddonUpdate = (
|
|
||||||
element: HTMLElement,
|
|
||||||
dialogParams: SupervisorDialogSupervisorAddonUpdateParams
|
|
||||||
): void => {
|
|
||||||
fireEvent(element, "show-dialog", {
|
|
||||||
dialogTag: "dialog-supervisor-addon-update",
|
|
||||||
dialogImport: () => import("./dialog-supervisor-addon-update"),
|
|
||||||
dialogParams,
|
|
||||||
});
|
|
||||||
};
|
|
@@ -1,175 +0,0 @@
|
|||||||
import "@material/mwc-button/mwc-button";
|
|
||||||
import {
|
|
||||||
css,
|
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
|
||||||
import "../../../../src/components/ha-circular-progress";
|
|
||||||
import "../../../../src/components/ha-dialog";
|
|
||||||
import "../../../../src/components/ha-settings-row";
|
|
||||||
import "../../../../src/components/ha-svg-icon";
|
|
||||||
import "../../../../src/components/ha-switch";
|
|
||||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
|
||||||
import { createHassioPartialSnapshot } from "../../../../src/data/hassio/snapshot";
|
|
||||||
import { HassioHomeAssistantInfo } from "../../../../src/data/hassio/supervisor";
|
|
||||||
import { updateCore } from "../../../../src/data/supervisor/core";
|
|
||||||
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
|
||||||
import type { HomeAssistant } from "../../../../src/types";
|
|
||||||
import { SupervisorDialogSupervisorCoreUpdateParams } from "./show-dialog-core-update";
|
|
||||||
|
|
||||||
@customElement("dialog-supervisor-core-update")
|
|
||||||
class DialogSupervisorCoreUpdate extends LitElement {
|
|
||||||
public hass!: HomeAssistant;
|
|
||||||
|
|
||||||
public core!: HassioHomeAssistantInfo;
|
|
||||||
|
|
||||||
@internalProperty() private _opened = false;
|
|
||||||
|
|
||||||
@internalProperty() private _createSnapshot = true;
|
|
||||||
|
|
||||||
@internalProperty() private _action: "snapshot" | "update" | null = null;
|
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
|
||||||
|
|
||||||
public async showDialog(
|
|
||||||
params: SupervisorDialogSupervisorCoreUpdateParams
|
|
||||||
): Promise<void> {
|
|
||||||
this._opened = true;
|
|
||||||
this.core = params.core;
|
|
||||||
await this.updateComplete;
|
|
||||||
}
|
|
||||||
|
|
||||||
public closeDialog(): void {
|
|
||||||
this._action = null;
|
|
||||||
this._createSnapshot = true;
|
|
||||||
this._opened = false;
|
|
||||||
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
|
||||||
}
|
|
||||||
|
|
||||||
public focus(): void {
|
|
||||||
this.updateComplete.then(() =>
|
|
||||||
(this.shadowRoot?.querySelector(
|
|
||||||
"[dialogInitialFocus]"
|
|
||||||
) as HTMLElement)?.focus()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
|
||||||
return html`
|
|
||||||
<ha-dialog .open=${this._opened} scrimClickAction escapeKeyAction>
|
|
||||||
${this._action === null
|
|
||||||
? html`<slot name="heading">
|
|
||||||
<h2 id="title" class="header_title">
|
|
||||||
Update Home Assistant Core
|
|
||||||
</h2>
|
|
||||||
</slot>
|
|
||||||
<div>
|
|
||||||
Are you sure you want to update Home Assistant Core to version
|
|
||||||
${this.core.version_latest}?
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<ha-settings-row three-rows>
|
|
||||||
<span slot="heading">
|
|
||||||
Snapshot
|
|
||||||
</span>
|
|
||||||
<span slot="description">
|
|
||||||
Create a snapshot of Home Assistant Core before updating
|
|
||||||
</span>
|
|
||||||
<ha-switch
|
|
||||||
.checked=${this._createSnapshot}
|
|
||||||
haptic
|
|
||||||
title="Create snapshot"
|
|
||||||
@click=${this._toggleSnapshot}
|
|
||||||
>
|
|
||||||
</ha-switch>
|
|
||||||
</ha-settings-row>
|
|
||||||
<mwc-button @click=${this.closeDialog} slot="secondaryAction">
|
|
||||||
Cancel
|
|
||||||
</mwc-button>
|
|
||||||
<mwc-button @click=${this._update} slot="primaryAction">
|
|
||||||
Update
|
|
||||||
</mwc-button>`
|
|
||||||
: html`<ha-circular-progress alt="Updating" size="large" active>
|
|
||||||
</ha-circular-progress>
|
|
||||||
<p class="progress-text">
|
|
||||||
${this._action === "update"
|
|
||||||
? `Updating Home Assistant Core to version ${this.core.version_latest}`
|
|
||||||
: "Creating snapshot of Home Assistant Core"}
|
|
||||||
</p>`}
|
|
||||||
${this._error ? html`<p class="error">${this._error}</p>` : ""}
|
|
||||||
</ha-dialog>
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
|
|
||||||
private _toggleSnapshot() {
|
|
||||||
this._createSnapshot = !this._createSnapshot;
|
|
||||||
}
|
|
||||||
|
|
||||||
private async _update() {
|
|
||||||
if (this._createSnapshot) {
|
|
||||||
this._action = "snapshot";
|
|
||||||
try {
|
|
||||||
await createHassioPartialSnapshot(this.hass, {
|
|
||||||
name: `core_${this.core.version}`,
|
|
||||||
folders: ["homeassistant"],
|
|
||||||
homeassistant: true,
|
|
||||||
});
|
|
||||||
} catch (err) {
|
|
||||||
this._error = extractApiErrorMessage(err);
|
|
||||||
this._action = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
this._action = "update";
|
|
||||||
try {
|
|
||||||
await updateCore(this.hass);
|
|
||||||
} catch (err) {
|
|
||||||
if (this.hass.connection.connected) {
|
|
||||||
this._error = extractApiErrorMessage(err);
|
|
||||||
this._action = null;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fireEvent(this, "supervisor-colllection-refresh", { colllection: "core" });
|
|
||||||
this.closeDialog();
|
|
||||||
}
|
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
|
||||||
return [
|
|
||||||
haStyle,
|
|
||||||
haStyleDialog,
|
|
||||||
css`
|
|
||||||
.form {
|
|
||||||
color: var(--primary-text-color);
|
|
||||||
}
|
|
||||||
|
|
||||||
ha-settings-row {
|
|
||||||
margin-top: 32px;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
ha-circular-progress {
|
|
||||||
display: block;
|
|
||||||
margin: 32px;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.progress-text {
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
declare global {
|
|
||||||
interface HTMLElementTagNameMap {
|
|
||||||
"dialog-supervisor-core-update": DialogSupervisorCoreUpdate;
|
|
||||||
}
|
|
||||||
}
|
|
@@ -1,17 +0,0 @@
|
|||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
|
||||||
import { HassioHomeAssistantInfo } from "../../../../src/data/hassio/supervisor";
|
|
||||||
|
|
||||||
export interface SupervisorDialogSupervisorCoreUpdateParams {
|
|
||||||
core: HassioHomeAssistantInfo;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const showDialogSupervisorCoreUpdate = (
|
|
||||||
element: HTMLElement,
|
|
||||||
dialogParams: SupervisorDialogSupervisorCoreUpdateParams
|
|
||||||
): void => {
|
|
||||||
fireEvent(element, "show-dialog", {
|
|
||||||
dialogTag: "dialog-supervisor-core-update",
|
|
||||||
dialogImport: () => import("./dialog-supervisor-core-update"),
|
|
||||||
dialogParams,
|
|
||||||
});
|
|
||||||
};
|
|
194
hassio/src/dialogs/hardware/dialog-hassio-hardware.ts
Executable file
194
hassio/src/dialogs/hardware/dialog-hassio-hardware.ts
Executable file
@@ -0,0 +1,194 @@
|
|||||||
|
import { mdiClose } from "@mdi/js";
|
||||||
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement, property, state } from "lit/decorators";
|
||||||
|
import memoizeOne from "memoize-one";
|
||||||
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
|
import "../../../../src/common/search/search-input";
|
||||||
|
import { compare } from "../../../../src/common/string/compare";
|
||||||
|
import "../../../../src/components/ha-dialog";
|
||||||
|
import "../../../../src/components/ha-expansion-panel";
|
||||||
|
import { HassioHardwareInfo } from "../../../../src/data/hassio/hardware";
|
||||||
|
import { dump } from "../../../../src/resources/js-yaml-dump";
|
||||||
|
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
||||||
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
|
import { HassioHardwareDialogParams } from "./show-dialog-hassio-hardware";
|
||||||
|
|
||||||
|
const _filterDevices = memoizeOne(
|
||||||
|
(showAdvanced: boolean, hardware: HassioHardwareInfo, filter: string) =>
|
||||||
|
hardware.devices
|
||||||
|
.filter(
|
||||||
|
(device) =>
|
||||||
|
(showAdvanced ||
|
||||||
|
["tty", "gpio", "input"].includes(device.subsystem)) &&
|
||||||
|
(device.by_id?.toLowerCase().includes(filter) ||
|
||||||
|
device.name.toLowerCase().includes(filter) ||
|
||||||
|
device.dev_path.toLocaleLowerCase().includes(filter) ||
|
||||||
|
JSON.stringify(device.attributes)
|
||||||
|
.toLocaleLowerCase()
|
||||||
|
.includes(filter))
|
||||||
|
)
|
||||||
|
.sort((a, b) => compare(a.name, b.name))
|
||||||
|
);
|
||||||
|
|
||||||
|
@customElement("dialog-hassio-hardware")
|
||||||
|
class HassioHardwareDialog extends LitElement {
|
||||||
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@state() private _dialogParams?: HassioHardwareDialogParams;
|
||||||
|
|
||||||
|
@state() private _filter?: string;
|
||||||
|
|
||||||
|
public showDialog(params: HassioHardwareDialogParams) {
|
||||||
|
this._dialogParams = params;
|
||||||
|
}
|
||||||
|
|
||||||
|
public closeDialog() {
|
||||||
|
this._dialogParams = undefined;
|
||||||
|
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
||||||
|
}
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this._dialogParams) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
|
||||||
|
const devices = _filterDevices(
|
||||||
|
this.hass.userData?.showAdvanced || false,
|
||||||
|
this._dialogParams.hardware,
|
||||||
|
(this._filter || "").toLowerCase()
|
||||||
|
);
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<ha-dialog
|
||||||
|
open
|
||||||
|
scrimClickAction
|
||||||
|
hideActions
|
||||||
|
@closed=${this.closeDialog}
|
||||||
|
.heading=${true}
|
||||||
|
>
|
||||||
|
<div class="header" slot="heading">
|
||||||
|
<h2>
|
||||||
|
${this._dialogParams.supervisor.localize("dialog.hardware.title")}
|
||||||
|
</h2>
|
||||||
|
<mwc-icon-button dialogAction="close">
|
||||||
|
<ha-svg-icon .path=${mdiClose}></ha-svg-icon>
|
||||||
|
</mwc-icon-button>
|
||||||
|
<search-input
|
||||||
|
autofocus
|
||||||
|
no-label-float
|
||||||
|
.filter=${this._filter}
|
||||||
|
@value-changed=${this._handleSearchChange}
|
||||||
|
.label=${this._dialogParams.supervisor.localize(
|
||||||
|
"dialog.hardware.search"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
</search-input>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
${devices.map(
|
||||||
|
(device) =>
|
||||||
|
html`<ha-expansion-panel
|
||||||
|
.header=${device.name}
|
||||||
|
.secondary=${device.by_id || undefined}
|
||||||
|
outlined
|
||||||
|
>
|
||||||
|
<div class="device-property">
|
||||||
|
<span>
|
||||||
|
${this._dialogParams!.supervisor.localize(
|
||||||
|
"dialog.hardware.subsystem"
|
||||||
|
)}:
|
||||||
|
</span>
|
||||||
|
<span>${device.subsystem}</span>
|
||||||
|
</div>
|
||||||
|
<div class="device-property">
|
||||||
|
<span>
|
||||||
|
${this._dialogParams!.supervisor.localize(
|
||||||
|
"dialog.hardware.device_path"
|
||||||
|
)}:
|
||||||
|
</span>
|
||||||
|
<code>${device.dev_path}</code>
|
||||||
|
</div>
|
||||||
|
${device.by_id
|
||||||
|
? html` <div class="device-property">
|
||||||
|
<span>
|
||||||
|
${this._dialogParams!.supervisor.localize(
|
||||||
|
"dialog.hardware.id"
|
||||||
|
)}:
|
||||||
|
</span>
|
||||||
|
<code>${device.by_id}</code>
|
||||||
|
</div>`
|
||||||
|
: ""}
|
||||||
|
<div class="attributes">
|
||||||
|
<span>
|
||||||
|
${this._dialogParams!.supervisor.localize(
|
||||||
|
"dialog.hardware.attributes"
|
||||||
|
)}:
|
||||||
|
</span>
|
||||||
|
<pre>${dump(device.attributes, { indent: 2 })}</pre>
|
||||||
|
</div>
|
||||||
|
</ha-expansion-panel>`
|
||||||
|
)}
|
||||||
|
</ha-dialog>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleSearchChange(ev: CustomEvent) {
|
||||||
|
this._filter = ev.detail.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResultGroup {
|
||||||
|
return [
|
||||||
|
haStyle,
|
||||||
|
haStyleDialog,
|
||||||
|
css`
|
||||||
|
mwc-icon-button {
|
||||||
|
position: absolute;
|
||||||
|
right: 16px;
|
||||||
|
top: 10px;
|
||||||
|
text-decoration: none;
|
||||||
|
color: var(--primary-text-color);
|
||||||
|
}
|
||||||
|
h2 {
|
||||||
|
margin: 18px 42px 0 18px;
|
||||||
|
color: var(--primary-text-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
ha-expansion-panel {
|
||||||
|
margin: 4px 0;
|
||||||
|
}
|
||||||
|
pre,
|
||||||
|
code {
|
||||||
|
background-color: var(--markdown-code-background-color, none);
|
||||||
|
border-radius: 3px;
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
padding: 16px;
|
||||||
|
overflow: auto;
|
||||||
|
line-height: 1.45;
|
||||||
|
font-family: var(--code-font-family, monospace);
|
||||||
|
}
|
||||||
|
code {
|
||||||
|
font-size: 85%;
|
||||||
|
padding: 0.2em 0.4em;
|
||||||
|
}
|
||||||
|
search-input {
|
||||||
|
margin: 0 16px;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
.device-property {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
.attributes {
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"dialog-hassio-hardware": HassioHardwareDialog;
|
||||||
|
}
|
||||||
|
}
|
19
hassio/src/dialogs/hardware/show-dialog-hassio-hardware.ts
Normal file
19
hassio/src/dialogs/hardware/show-dialog-hassio-hardware.ts
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
|
import { HassioHardwareInfo } from "../../../../src/data/hassio/hardware";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
|
|
||||||
|
export interface HassioHardwareDialogParams {
|
||||||
|
supervisor: Supervisor;
|
||||||
|
hardware: HassioHardwareInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const showHassioHardwareDialog = (
|
||||||
|
element: HTMLElement,
|
||||||
|
dialogParams: HassioHardwareDialogParams
|
||||||
|
): void => {
|
||||||
|
fireEvent(element, "show-dialog", {
|
||||||
|
dialogTag: "dialog-hassio-hardware",
|
||||||
|
dialogImport: () => import("./dialog-hassio-hardware"),
|
||||||
|
dialogParams,
|
||||||
|
});
|
||||||
|
};
|
@@ -1,13 +1,5 @@
|
|||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property, state } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
||||||
import "../../../../src/components/ha-markdown";
|
import "../../../../src/components/ha-markdown";
|
||||||
import { haStyleDialog } from "../../../../src/resources/styles";
|
import { haStyleDialog } from "../../../../src/resources/styles";
|
||||||
@@ -23,7 +15,7 @@ class HassioMarkdownDialog extends LitElement {
|
|||||||
|
|
||||||
@property() public content!: string;
|
@property() public content!: string;
|
||||||
|
|
||||||
@internalProperty() private _opened = false;
|
@state() private _opened = false;
|
||||||
|
|
||||||
public showDialog(params: HassioMarkdownDialogParams) {
|
public showDialog(params: HassioMarkdownDialogParams) {
|
||||||
this.title = params.title;
|
this.title = params.title;
|
||||||
@@ -50,7 +42,7 @@ class HassioMarkdownDialog extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyleDialog,
|
haStyleDialog,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
|
@@ -6,19 +6,10 @@ import "@material/mwc-tab";
|
|||||||
import "@material/mwc-tab-bar";
|
import "@material/mwc-tab-bar";
|
||||||
import { mdiClose } from "@mdi/js";
|
import { mdiClose } from "@mdi/js";
|
||||||
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property, state } from "lit/decorators";
|
||||||
CSSResult,
|
import { cache } from "lit/directives/cache";
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import { cache } from "lit-html/directives/cache";
|
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import "../../../../src/components/ha-chips";
|
|
||||||
import "../../../../src/components/ha-circular-progress";
|
import "../../../../src/components/ha-circular-progress";
|
||||||
import "../../../../src/components/ha-dialog";
|
import "../../../../src/components/ha-dialog";
|
||||||
import "../../../../src/components/ha-expansion-panel";
|
import "../../../../src/components/ha-expansion-panel";
|
||||||
@@ -35,6 +26,7 @@ import {
|
|||||||
updateNetworkInterface,
|
updateNetworkInterface,
|
||||||
WifiConfiguration,
|
WifiConfiguration,
|
||||||
} from "../../../../src/data/hassio/network";
|
} from "../../../../src/data/hassio/network";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import {
|
import {
|
||||||
showAlertDialog,
|
showAlertDialog,
|
||||||
showConfirmationDialog,
|
showConfirmationDialog,
|
||||||
@@ -47,35 +39,39 @@ import { HassioNetworkDialogParams } from "./show-dialog-network";
|
|||||||
const IP_VERSIONS = ["ipv4", "ipv6"];
|
const IP_VERSIONS = ["ipv4", "ipv6"];
|
||||||
|
|
||||||
@customElement("dialog-hassio-network")
|
@customElement("dialog-hassio-network")
|
||||||
export class DialogHassioNetwork extends LitElement
|
export class DialogHassioNetwork
|
||||||
|
extends LitElement
|
||||||
implements HassDialog<HassioNetworkDialogParams> {
|
implements HassDialog<HassioNetworkDialogParams> {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@internalProperty() private _accessPoints?: AccessPoints;
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@internalProperty() private _curTabIndex = 0;
|
@state() private _accessPoints?: AccessPoints;
|
||||||
|
|
||||||
@internalProperty() private _dirty = false;
|
@state() private _curTabIndex = 0;
|
||||||
|
|
||||||
@internalProperty() private _interface?: NetworkInterface;
|
@state() private _dirty = false;
|
||||||
|
|
||||||
@internalProperty() private _interfaces!: NetworkInterface[];
|
@state() private _interface?: NetworkInterface;
|
||||||
|
|
||||||
@internalProperty() private _params?: HassioNetworkDialogParams;
|
@state() private _interfaces!: NetworkInterface[];
|
||||||
|
|
||||||
@internalProperty() private _processing = false;
|
@state() private _params?: HassioNetworkDialogParams;
|
||||||
|
|
||||||
@internalProperty() private _scanning = false;
|
@state() private _processing = false;
|
||||||
|
|
||||||
@internalProperty() private _wifiConfiguration?: WifiConfiguration;
|
@state() private _scanning = false;
|
||||||
|
|
||||||
|
@state() private _wifiConfiguration?: WifiConfiguration;
|
||||||
|
|
||||||
public async showDialog(params: HassioNetworkDialogParams): Promise<void> {
|
public async showDialog(params: HassioNetworkDialogParams): Promise<void> {
|
||||||
this._params = params;
|
this._params = params;
|
||||||
this._dirty = false;
|
this._dirty = false;
|
||||||
this._curTabIndex = 0;
|
this._curTabIndex = 0;
|
||||||
this._interfaces = params.network.interfaces.sort((a, b) => {
|
this.supervisor = params.supervisor;
|
||||||
return a.primary > b.primary ? -1 : 1;
|
this._interfaces = params.supervisor.network.interfaces.sort((a, b) =>
|
||||||
});
|
a.primary > b.primary ? -1 : 1
|
||||||
|
);
|
||||||
this._interface = { ...this._interfaces[this._curTabIndex] };
|
this._interface = { ...this._interfaces[this._curTabIndex] };
|
||||||
|
|
||||||
await this.updateComplete;
|
await this.updateComplete;
|
||||||
@@ -104,7 +100,7 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
<div slot="heading">
|
<div slot="heading">
|
||||||
<ha-header-bar>
|
<ha-header-bar>
|
||||||
<span slot="title">
|
<span slot="title">
|
||||||
Network settings
|
${this.supervisor.localize("dialog.network.title")}
|
||||||
</span>
|
</span>
|
||||||
<mwc-icon-button slot="actionItems" dialogAction="cancel">
|
<mwc-icon-button slot="actionItems" dialogAction="cancel">
|
||||||
<ha-svg-icon .path=${mdiClose}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiClose}></ha-svg-icon>
|
||||||
@@ -139,7 +135,13 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
? html`
|
? html`
|
||||||
<ha-expansion-panel header="Wi-Fi" outlined>
|
<ha-expansion-panel header="Wi-Fi" outlined>
|
||||||
${this._interface?.wifi?.ssid
|
${this._interface?.wifi?.ssid
|
||||||
? html`<p>Connected to: ${this._interface?.wifi?.ssid}</p>`
|
? html`<p>
|
||||||
|
${this.supervisor.localize(
|
||||||
|
"dialog.network.connected_to",
|
||||||
|
"ssid",
|
||||||
|
this._interface?.wifi?.ssid
|
||||||
|
)}
|
||||||
|
</p>`
|
||||||
: ""}
|
: ""}
|
||||||
<mwc-button
|
<mwc-button
|
||||||
class="scan"
|
class="scan"
|
||||||
@@ -149,7 +151,7 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
${this._scanning
|
${this._scanning
|
||||||
? html`<ha-circular-progress active size="small">
|
? html`<ha-circular-progress active size="small">
|
||||||
</ha-circular-progress>`
|
</ha-circular-progress>`
|
||||||
: "Scan for accesspoints"}
|
: this.supervisor.localize("dialog.network.scan_ap")}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
${this._accessPoints &&
|
${this._accessPoints &&
|
||||||
this._accessPoints.accesspoints &&
|
this._accessPoints.accesspoints &&
|
||||||
@@ -181,7 +183,11 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
${this._wifiConfiguration
|
${this._wifiConfiguration
|
||||||
? html`
|
? html`
|
||||||
<div class="radio-row">
|
<div class="radio-row">
|
||||||
<ha-formfield label="open">
|
<ha-formfield
|
||||||
|
.label=${this.supervisor.localize(
|
||||||
|
"dialog.network.open"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<ha-radio
|
<ha-radio
|
||||||
@change=${this._handleRadioValueChangedAp}
|
@change=${this._handleRadioValueChangedAp}
|
||||||
.ap=${this._wifiConfiguration}
|
.ap=${this._wifiConfiguration}
|
||||||
@@ -193,7 +199,11 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
>
|
>
|
||||||
</ha-radio>
|
</ha-radio>
|
||||||
</ha-formfield>
|
</ha-formfield>
|
||||||
<ha-formfield label="wep">
|
<ha-formfield
|
||||||
|
.label=${this.supervisor.localize(
|
||||||
|
"dialog.network.wep"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<ha-radio
|
<ha-radio
|
||||||
@change=${this._handleRadioValueChangedAp}
|
@change=${this._handleRadioValueChangedAp}
|
||||||
.ap=${this._wifiConfiguration}
|
.ap=${this._wifiConfiguration}
|
||||||
@@ -203,7 +213,11 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
>
|
>
|
||||||
</ha-radio>
|
</ha-radio>
|
||||||
</ha-formfield>
|
</ha-formfield>
|
||||||
<ha-formfield label="wpa-psk">
|
<ha-formfield
|
||||||
|
.label=${this.supervisor.localize(
|
||||||
|
"dialog.network.wpa"
|
||||||
|
)}
|
||||||
|
>
|
||||||
<ha-radio
|
<ha-radio
|
||||||
@change=${this._handleRadioValueChangedAp}
|
@change=${this._handleRadioValueChangedAp}
|
||||||
.ap=${this._wifiConfiguration}
|
.ap=${this._wifiConfiguration}
|
||||||
@@ -237,18 +251,21 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
: ""}
|
: ""}
|
||||||
${this._dirty
|
${this._dirty
|
||||||
? html`<div class="warning">
|
? html`<div class="warning">
|
||||||
If you are changing the Wi-Fi, IP or gateway addresses, you might
|
${this.supervisor.localize("dialog.network.warning")}
|
||||||
lose the connection!
|
|
||||||
</div>`
|
</div>`
|
||||||
: ""}
|
: ""}
|
||||||
</div>
|
</div>
|
||||||
<div class="buttons">
|
<div class="buttons">
|
||||||
<mwc-button label="close" @click=${this.closeDialog}> </mwc-button>
|
<mwc-button
|
||||||
|
.label=${this.supervisor.localize("common.cancel")}
|
||||||
|
@click=${this.closeDialog}
|
||||||
|
>
|
||||||
|
</mwc-button>
|
||||||
<mwc-button @click=${this._updateNetwork} .disabled=${!this._dirty}>
|
<mwc-button @click=${this._updateNetwork} .disabled=${!this._dirty}>
|
||||||
${this._processing
|
${this._processing
|
||||||
? html`<ha-circular-progress active size="small">
|
? html`<ha-circular-progress active size="small">
|
||||||
</ha-circular-progress>`
|
</ha-circular-progress>`
|
||||||
: "Save"}
|
: this.supervisor.localize("common.save")}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
</div>`;
|
</div>`;
|
||||||
}
|
}
|
||||||
@@ -285,7 +302,9 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
outlined
|
outlined
|
||||||
>
|
>
|
||||||
<div class="radio-row">
|
<div class="radio-row">
|
||||||
<ha-formfield label="DHCP">
|
<ha-formfield
|
||||||
|
.label=${this.supervisor.localize("dialog.network.dhcp")}
|
||||||
|
>
|
||||||
<ha-radio
|
<ha-radio
|
||||||
@change=${this._handleRadioValueChanged}
|
@change=${this._handleRadioValueChanged}
|
||||||
.version=${version}
|
.version=${version}
|
||||||
@@ -295,7 +314,9 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
>
|
>
|
||||||
</ha-radio>
|
</ha-radio>
|
||||||
</ha-formfield>
|
</ha-formfield>
|
||||||
<ha-formfield label="Static">
|
<ha-formfield
|
||||||
|
.label=${this.supervisor.localize("dialog.network.static")}
|
||||||
|
>
|
||||||
<ha-radio
|
<ha-radio
|
||||||
@change=${this._handleRadioValueChanged}
|
@change=${this._handleRadioValueChanged}
|
||||||
.version=${version}
|
.version=${version}
|
||||||
@@ -305,7 +326,10 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
>
|
>
|
||||||
</ha-radio>
|
</ha-radio>
|
||||||
</ha-formfield>
|
</ha-formfield>
|
||||||
<ha-formfield label="Disabled" class="warning">
|
<ha-formfield
|
||||||
|
.label=${this.supervisor.localize("dialog.network.disabled")}
|
||||||
|
class="warning"
|
||||||
|
>
|
||||||
<ha-radio
|
<ha-radio
|
||||||
@change=${this._handleRadioValueChanged}
|
@change=${this._handleRadioValueChanged}
|
||||||
.version=${version}
|
.version=${version}
|
||||||
@@ -321,7 +345,7 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
<paper-input
|
<paper-input
|
||||||
class="flex-auto"
|
class="flex-auto"
|
||||||
id="address"
|
id="address"
|
||||||
label="IP address/Netmask"
|
.label=${this.supervisor.localize("dialog.network.ip_netmask")}
|
||||||
.version=${version}
|
.version=${version}
|
||||||
.value=${this._toString(this._interface![version].address)}
|
.value=${this._toString(this._interface![version].address)}
|
||||||
@value-changed=${this._handleInputValueChanged}
|
@value-changed=${this._handleInputValueChanged}
|
||||||
@@ -330,7 +354,7 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
<paper-input
|
<paper-input
|
||||||
class="flex-auto"
|
class="flex-auto"
|
||||||
id="gateway"
|
id="gateway"
|
||||||
label="Gateway address"
|
.label=${this.supervisor.localize("dialog.network.gateway")}
|
||||||
.version=${version}
|
.version=${version}
|
||||||
.value=${this._interface![version].gateway}
|
.value=${this._interface![version].gateway}
|
||||||
@value-changed=${this._handleInputValueChanged}
|
@value-changed=${this._handleInputValueChanged}
|
||||||
@@ -339,7 +363,7 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
<paper-input
|
<paper-input
|
||||||
class="flex-auto"
|
class="flex-auto"
|
||||||
id="nameservers"
|
id="nameservers"
|
||||||
label="DNS servers"
|
.label=${this.supervisor.localize("dialog.network.dns_servers")}
|
||||||
.version=${version}
|
.version=${version}
|
||||||
.value=${this._toString(this._interface![version].nameservers)}
|
.value=${this._toString(this._interface![version].nameservers)}
|
||||||
@value-changed=${this._handleInputValueChanged}
|
@value-changed=${this._handleInputValueChanged}
|
||||||
@@ -424,7 +448,7 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
);
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to change network settings",
|
title: this.supervisor.localize("dialog.network.failed_to_change"),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
this._processing = false;
|
this._processing = false;
|
||||||
@@ -437,10 +461,9 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
private async _handleTabActivated(ev: CustomEvent): Promise<void> {
|
private async _handleTabActivated(ev: CustomEvent): Promise<void> {
|
||||||
if (this._dirty) {
|
if (this._dirty) {
|
||||||
const confirm = await showConfirmationDialog(this, {
|
const confirm = await showConfirmationDialog(this, {
|
||||||
text:
|
text: this.supervisor.localize("dialog.network.unsaved"),
|
||||||
"You have unsaved changes, these will get lost if you change tabs, do you want to continue?",
|
confirmText: this.supervisor.localize("common.yes"),
|
||||||
confirmText: "yes",
|
dismissText: this.supervisor.localize("common.no"),
|
||||||
dismissText: "no",
|
|
||||||
});
|
});
|
||||||
if (!confirm) {
|
if (!confirm) {
|
||||||
this.requestUpdate("_interface");
|
this.requestUpdate("_interface");
|
||||||
@@ -512,7 +535,7 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
this._wifiConfiguration![id] = value;
|
this._wifiConfiguration![id] = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyleDialog,
|
haStyleDialog,
|
||||||
css`
|
css`
|
||||||
|
@@ -1,9 +1,9 @@
|
|||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import { NetworkInfo } from "../../../../src/data/hassio/network";
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import "./dialog-hassio-network";
|
import "./dialog-hassio-network";
|
||||||
|
|
||||||
export interface HassioNetworkDialogParams {
|
export interface HassioNetworkDialogParams {
|
||||||
network: NetworkInfo;
|
supervisor: Supervisor;
|
||||||
loadData: () => Promise<void>;
|
loadData: () => Promise<void>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -3,16 +3,8 @@ import "@material/mwc-icon-button/mwc-icon-button";
|
|||||||
import "@material/mwc-list/mwc-list-item";
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import { mdiDelete } from "@mdi/js";
|
import { mdiDelete } from "@mdi/js";
|
||||||
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property, state } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import "../../../../src/components/ha-circular-progress";
|
import "../../../../src/components/ha-circular-progress";
|
||||||
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
||||||
import "../../../../src/components/ha-svg-icon";
|
import "../../../../src/components/ha-svg-icon";
|
||||||
@@ -22,41 +14,45 @@ import {
|
|||||||
fetchHassioDockerRegistries,
|
fetchHassioDockerRegistries,
|
||||||
removeHassioDockerRegistry,
|
removeHassioDockerRegistry,
|
||||||
} from "../../../../src/data/hassio/docker";
|
} from "../../../../src/data/hassio/docker";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import { showAlertDialog } from "../../../../src/dialogs/generic/show-dialog-box";
|
import { showAlertDialog } from "../../../../src/dialogs/generic/show-dialog-box";
|
||||||
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
||||||
import type { HomeAssistant } from "../../../../src/types";
|
import type { HomeAssistant } from "../../../../src/types";
|
||||||
|
import { RegistriesDialogParams } from "./show-dialog-registries";
|
||||||
|
|
||||||
@customElement("dialog-hassio-registries")
|
@customElement("dialog-hassio-registries")
|
||||||
class HassioRegistriesDialog extends LitElement {
|
class HassioRegistriesDialog extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) private _registries?: {
|
@property({ attribute: false }) private _registries?: {
|
||||||
registry: string;
|
registry: string;
|
||||||
username: string;
|
username: string;
|
||||||
}[];
|
}[];
|
||||||
|
|
||||||
@internalProperty() private _registry?: string;
|
@state() private _registry?: string;
|
||||||
|
|
||||||
@internalProperty() private _username?: string;
|
@state() private _username?: string;
|
||||||
|
|
||||||
@internalProperty() private _password?: string;
|
@state() private _password?: string;
|
||||||
|
|
||||||
@internalProperty() private _opened = false;
|
@state() private _opened = false;
|
||||||
|
|
||||||
@internalProperty() private _addingRegistry = false;
|
@state() private _addingRegistry = false;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
<ha-dialog
|
<ha-dialog
|
||||||
.open=${this._opened}
|
.open=${this._opened}
|
||||||
@closing=${this.closeDialog}
|
@closed=${this.closeDialog}
|
||||||
scrimClickAction
|
scrimClickAction
|
||||||
escapeKeyAction
|
escapeKeyAction
|
||||||
.heading=${createCloseHeading(
|
.heading=${createCloseHeading(
|
||||||
this.hass,
|
this.hass,
|
||||||
this._addingRegistry
|
this._addingRegistry
|
||||||
? "Add New Docker Registry"
|
? this.supervisor.localize("dialog.registries.title_add")
|
||||||
: "Manage Docker Registries"
|
: this.supervisor.localize("dialog.registries.title_manage")
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
<div class="form">
|
<div class="form">
|
||||||
@@ -66,7 +62,9 @@ class HassioRegistriesDialog extends LitElement {
|
|||||||
@value-changed=${this._inputChanged}
|
@value-changed=${this._inputChanged}
|
||||||
class="flex-auto"
|
class="flex-auto"
|
||||||
name="registry"
|
name="registry"
|
||||||
label="Registry"
|
.label=${this.supervisor.localize(
|
||||||
|
"dialog.registries.registry"
|
||||||
|
)}
|
||||||
required
|
required
|
||||||
auto-validate
|
auto-validate
|
||||||
></paper-input>
|
></paper-input>
|
||||||
@@ -74,7 +72,9 @@ class HassioRegistriesDialog extends LitElement {
|
|||||||
@value-changed=${this._inputChanged}
|
@value-changed=${this._inputChanged}
|
||||||
class="flex-auto"
|
class="flex-auto"
|
||||||
name="username"
|
name="username"
|
||||||
label="Username"
|
.label=${this.supervisor.localize(
|
||||||
|
"dialog.registries.username"
|
||||||
|
)}
|
||||||
required
|
required
|
||||||
auto-validate
|
auto-validate
|
||||||
></paper-input>
|
></paper-input>
|
||||||
@@ -82,7 +82,9 @@ class HassioRegistriesDialog extends LitElement {
|
|||||||
@value-changed=${this._inputChanged}
|
@value-changed=${this._inputChanged}
|
||||||
class="flex-auto"
|
class="flex-auto"
|
||||||
name="password"
|
name="password"
|
||||||
label="Password"
|
.label=${this.supervisor.localize(
|
||||||
|
"dialog.registries.password"
|
||||||
|
)}
|
||||||
type="password"
|
type="password"
|
||||||
required
|
required
|
||||||
auto-validate
|
auto-validate
|
||||||
@@ -94,35 +96,46 @@ class HassioRegistriesDialog extends LitElement {
|
|||||||
)}
|
)}
|
||||||
@click=${this._addNewRegistry}
|
@click=${this._addNewRegistry}
|
||||||
>
|
>
|
||||||
Add registry
|
${this.supervisor.localize("dialog.registries.add_registry")}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
`
|
`
|
||||||
: html`${this._registries?.length
|
: html`${this._registries?.length
|
||||||
? this._registries.map((entry) => {
|
? this._registries.map(
|
||||||
return html`
|
(entry) => html`
|
||||||
<mwc-list-item class="option" hasMeta twoline>
|
<mwc-list-item class="option" hasMeta twoline>
|
||||||
<span>${entry.registry}</span>
|
<span>${entry.registry}</span>
|
||||||
<span slot="secondary"
|
<span slot="secondary"
|
||||||
>Username: ${entry.username}</span
|
>${this.supervisor.localize(
|
||||||
|
"dialog.registries.username"
|
||||||
|
)}:
|
||||||
|
${entry.username}</span
|
||||||
>
|
>
|
||||||
<mwc-icon-button
|
<mwc-icon-button
|
||||||
.entry=${entry}
|
.entry=${entry}
|
||||||
title="Remove"
|
.title=${this.supervisor.localize(
|
||||||
|
"dialog.registries.remove"
|
||||||
|
)}
|
||||||
slot="meta"
|
slot="meta"
|
||||||
@click=${this._removeRegistry}
|
@click=${this._removeRegistry}
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiDelete}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiDelete}></ha-svg-icon>
|
||||||
</mwc-icon-button>
|
</mwc-icon-button>
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
`;
|
`
|
||||||
})
|
)
|
||||||
: html`
|
: html`
|
||||||
<mwc-list-item>
|
<mwc-list-item>
|
||||||
<span>No registries configured</span>
|
<span
|
||||||
|
>${this.supervisor.localize(
|
||||||
|
"dialog.registries.no_registries"
|
||||||
|
)}</span
|
||||||
|
>
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
`}
|
`}
|
||||||
<mwc-button @click=${this._addRegistry}>
|
<mwc-button @click=${this._addRegistry}>
|
||||||
Add new registry
|
${this.supervisor.localize(
|
||||||
|
"dialog.registries.add_new_registry"
|
||||||
|
)}
|
||||||
</mwc-button> `}
|
</mwc-button> `}
|
||||||
</div>
|
</div>
|
||||||
</ha-dialog>
|
</ha-dialog>
|
||||||
@@ -134,8 +147,9 @@ class HassioRegistriesDialog extends LitElement {
|
|||||||
this[`_${target.name}`] = target.value;
|
this[`_${target.name}`] = target.value;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async showDialog(_dialogParams: any): Promise<void> {
|
public async showDialog(dialogParams: RegistriesDialogParams): Promise<void> {
|
||||||
this._opened = true;
|
this._opened = true;
|
||||||
|
this.supervisor = dialogParams.supervisor;
|
||||||
await this._loadRegistries();
|
await this._loadRegistries();
|
||||||
await this.updateComplete;
|
await this.updateComplete;
|
||||||
}
|
}
|
||||||
@@ -178,7 +192,7 @@ class HassioRegistriesDialog extends LitElement {
|
|||||||
this._addingRegistry = false;
|
this._addingRegistry = false;
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to add registry",
|
title: this.supervisor.localize("dialog.registries.failed_to_add"),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -192,13 +206,13 @@ class HassioRegistriesDialog extends LitElement {
|
|||||||
await this._loadRegistries();
|
await this._loadRegistries();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to remove registry",
|
title: this.supervisor.localize("dialog.registries.failed_to_remove"),
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
haStyleDialog,
|
haStyleDialog,
|
||||||
@@ -230,9 +244,6 @@ class HassioRegistriesDialog extends LitElement {
|
|||||||
mwc-list-item span[slot="secondary"] {
|
mwc-list-item span[slot="secondary"] {
|
||||||
color: var(--secondary-text-color);
|
color: var(--secondary-text-color);
|
||||||
}
|
}
|
||||||
ha-paper-dropdown-menu {
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
@@ -1,10 +1,18 @@
|
|||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
|
import { Supervisor } from "../../../../src/data/supervisor/supervisor";
|
||||||
import "./dialog-hassio-registries";
|
import "./dialog-hassio-registries";
|
||||||
|
|
||||||
export const showRegistriesDialog = (element: HTMLElement): void => {
|
export interface RegistriesDialogParams {
|
||||||
|
supervisor: Supervisor;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const showRegistriesDialog = (
|
||||||
|
element: HTMLElement,
|
||||||
|
dialogParams: RegistriesDialogParams
|
||||||
|
): void => {
|
||||||
fireEvent(element, "show-dialog", {
|
fireEvent(element, "show-dialog", {
|
||||||
dialogTag: "dialog-hassio-registries",
|
dialogTag: "dialog-hassio-registries",
|
||||||
dialogImport: () => import("./dialog-hassio-registries"),
|
dialogImport: () => import("./dialog-hassio-registries"),
|
||||||
dialogParams: {},
|
dialogParams,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@@ -5,20 +5,12 @@ import "@polymer/paper-input/paper-input";
|
|||||||
import type { PaperInputElement } from "@polymer/paper-input/paper-input";
|
import type { PaperInputElement } from "@polymer/paper-input/paper-input";
|
||||||
import "@polymer/paper-item/paper-item";
|
import "@polymer/paper-item/paper-item";
|
||||||
import "@polymer/paper-item/paper-item-body";
|
import "@polymer/paper-item/paper-item-body";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
css,
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
CSSResult,
|
|
||||||
customElement,
|
|
||||||
html,
|
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
query,
|
|
||||||
TemplateResult,
|
|
||||||
} from "lit-element";
|
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import "../../../../src/components/ha-circular-progress";
|
import "../../../../src/components/ha-circular-progress";
|
||||||
import "../../../../src/components/ha-dialog";
|
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
||||||
import "../../../../src/components/ha-svg-icon";
|
import "../../../../src/components/ha-svg-icon";
|
||||||
import {
|
import {
|
||||||
fetchHassioAddonsInfo,
|
fetchHassioAddonsInfo,
|
||||||
@@ -34,27 +26,29 @@ import { HassioRepositoryDialogParams } from "./show-dialog-repositories";
|
|||||||
class HassioRepositoriesDialog extends LitElement {
|
class HassioRepositoriesDialog extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property({ attribute: false }) private _repos: HassioAddonRepository[] = [];
|
|
||||||
|
|
||||||
@property({ attribute: false })
|
|
||||||
private _dialogParams?: HassioRepositoryDialogParams;
|
|
||||||
|
|
||||||
@query("#repository_input", true) private _optionInput?: PaperInputElement;
|
@query("#repository_input", true) private _optionInput?: PaperInputElement;
|
||||||
|
|
||||||
@internalProperty() private _opened = false;
|
@state() private _repositories?: HassioAddonRepository[];
|
||||||
|
|
||||||
@internalProperty() private _prosessing = false;
|
@state() private _dialogParams?: HassioRepositoryDialogParams;
|
||||||
|
|
||||||
@internalProperty() private _error?: string;
|
@state() private _opened = false;
|
||||||
|
|
||||||
public async showDialog(_dialogParams: any): Promise<void> {
|
@state() private _processing = false;
|
||||||
this._dialogParams = _dialogParams;
|
|
||||||
this._repos = _dialogParams.repos;
|
@state() private _error?: string;
|
||||||
|
|
||||||
|
public async showDialog(
|
||||||
|
dialogParams: HassioRepositoryDialogParams
|
||||||
|
): Promise<void> {
|
||||||
|
this._dialogParams = dialogParams;
|
||||||
this._opened = true;
|
this._opened = true;
|
||||||
|
await this._loadData();
|
||||||
await this.updateComplete;
|
await this.updateComplete;
|
||||||
}
|
}
|
||||||
|
|
||||||
public closeDialog(): void {
|
public closeDialog(): void {
|
||||||
|
this._dialogParams = undefined;
|
||||||
this._opened = false;
|
this._opened = false;
|
||||||
this._error = "";
|
this._error = "";
|
||||||
}
|
}
|
||||||
@@ -66,20 +60,26 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
);
|
);
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
const repositories = this._filteredRepositories(this._repos);
|
if (!this._dialogParams?.supervisor || this._repositories === undefined) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
const repositories = this._filteredRepositories(this._repositories);
|
||||||
return html`
|
return html`
|
||||||
<ha-dialog
|
<ha-dialog
|
||||||
.open=${this._opened}
|
.open=${this._opened}
|
||||||
@closing=${this.closeDialog}
|
@closed=${this.closeDialog}
|
||||||
scrimClickAction
|
scrimClickAction
|
||||||
escapeKeyAction
|
escapeKeyAction
|
||||||
heading="Manage add-on repositories"
|
.heading=${createCloseHeading(
|
||||||
|
this.hass,
|
||||||
|
this._dialogParams!.supervisor.localize("dialog.repositories.title")
|
||||||
|
)}
|
||||||
>
|
>
|
||||||
${this._error ? html`<div class="error">${this._error}</div>` : ""}
|
${this._error ? html`<div class="error">${this._error}</div>` : ""}
|
||||||
<div class="form">
|
<div class="form">
|
||||||
${repositories.length
|
${repositories.length
|
||||||
? repositories.map((repo) => {
|
? repositories.map(
|
||||||
return html`
|
(repo) => html`
|
||||||
<paper-item class="option">
|
<paper-item class="option">
|
||||||
<paper-item-body three-line>
|
<paper-item-body three-line>
|
||||||
<div>${repo.name}</div>
|
<div>${repo.name}</div>
|
||||||
@@ -88,41 +88,47 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
</paper-item-body>
|
</paper-item-body>
|
||||||
<mwc-icon-button
|
<mwc-icon-button
|
||||||
.slug=${repo.slug}
|
.slug=${repo.slug}
|
||||||
title="Remove"
|
.title=${this._dialogParams!.supervisor.localize(
|
||||||
|
"dialog.repositories.remove"
|
||||||
|
)}
|
||||||
@click=${this._removeRepository}
|
@click=${this._removeRepository}
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${mdiDelete}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiDelete}></ha-svg-icon>
|
||||||
</mwc-icon-button>
|
</mwc-icon-button>
|
||||||
</paper-item>
|
</paper-item>
|
||||||
`;
|
`
|
||||||
})
|
)
|
||||||
: html`
|
: html` <paper-item> No repositories </paper-item> `}
|
||||||
<paper-item>
|
|
||||||
No repositories
|
|
||||||
</paper-item>
|
|
||||||
`}
|
|
||||||
<div class="layout horizontal bottom">
|
<div class="layout horizontal bottom">
|
||||||
<paper-input
|
<paper-input
|
||||||
class="flex-auto"
|
class="flex-auto"
|
||||||
id="repository_input"
|
id="repository_input"
|
||||||
label="Add repository"
|
.value=${this._dialogParams!.url || ""}
|
||||||
|
.label=${this._dialogParams!.supervisor.localize(
|
||||||
|
"dialog.repositories.add"
|
||||||
|
)}
|
||||||
@keydown=${this._handleKeyAdd}
|
@keydown=${this._handleKeyAdd}
|
||||||
></paper-input>
|
></paper-input>
|
||||||
<mwc-button @click=${this._addRepository}>
|
<mwc-button @click=${this._addRepository}>
|
||||||
${this._prosessing
|
${this._processing
|
||||||
? html`<ha-circular-progress active></ha-circular-progress>`
|
? html`<ha-circular-progress
|
||||||
: "Add"}
|
active
|
||||||
|
size="small"
|
||||||
|
></ha-circular-progress>`
|
||||||
|
: this._dialogParams!.supervisor.localize(
|
||||||
|
"dialog.repositories.add"
|
||||||
|
)}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<mwc-button slot="primaryAction" @click="${this.closeDialog}">
|
<mwc-button slot="primaryAction" @click=${this.closeDialog}>
|
||||||
Close
|
${this._dialogParams?.supervisor.localize("common.close")}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
</ha-dialog>
|
</ha-dialog>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResultGroup {
|
||||||
return [
|
return [
|
||||||
haStyle,
|
haStyle,
|
||||||
haStyleDialog,
|
haStyleDialog,
|
||||||
@@ -144,8 +150,10 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
mwc-button {
|
mwc-button {
|
||||||
margin-left: 8px;
|
margin-left: 8px;
|
||||||
}
|
}
|
||||||
ha-paper-dropdown-menu {
|
ha-circular-progress {
|
||||||
display: block;
|
display: block;
|
||||||
|
margin: 32px;
|
||||||
|
text-align: center;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
@@ -167,61 +175,57 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
this._addRepository();
|
this._addRepository();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async _loadData(): Promise<void> {
|
||||||
|
try {
|
||||||
|
const addonsinfo = await fetchHassioAddonsInfo(this.hass);
|
||||||
|
|
||||||
|
this._repositories = addonsinfo.repositories;
|
||||||
|
|
||||||
|
fireEvent(this, "supervisor-collection-refresh", { collection: "addon" });
|
||||||
|
} catch (err) {
|
||||||
|
this._error = extractApiErrorMessage(err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private async _addRepository() {
|
private async _addRepository() {
|
||||||
const input = this._optionInput;
|
const input = this._optionInput;
|
||||||
if (!input || !input.value) {
|
if (!input || !input.value) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this._prosessing = true;
|
this._processing = true;
|
||||||
const repositories = this._filteredRepositories(this._repos);
|
const repositories = this._filteredRepositories(this._repositories!);
|
||||||
const newRepositories = repositories.map((repo) => {
|
const newRepositories = repositories.map((repo) => repo.source);
|
||||||
return repo.source;
|
|
||||||
});
|
|
||||||
newRepositories.push(input.value);
|
newRepositories.push(input.value);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await setSupervisorOption(this.hass, {
|
await setSupervisorOption(this.hass, {
|
||||||
addons_repositories: newRepositories,
|
addons_repositories: newRepositories,
|
||||||
});
|
});
|
||||||
|
await this._loadData();
|
||||||
const addonsInfo = await fetchHassioAddonsInfo(this.hass);
|
|
||||||
this._repos = addonsInfo.repositories;
|
|
||||||
|
|
||||||
await this._dialogParams!.loadData();
|
|
||||||
|
|
||||||
input.value = "";
|
input.value = "";
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = extractApiErrorMessage(err);
|
this._error = extractApiErrorMessage(err);
|
||||||
}
|
}
|
||||||
this._prosessing = false;
|
this._processing = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _removeRepository(ev: Event) {
|
private async _removeRepository(ev: Event) {
|
||||||
const slug = (ev.currentTarget as any).slug;
|
const slug = (ev.currentTarget as any).slug;
|
||||||
const repositories = this._filteredRepositories(this._repos);
|
const repositories = this._filteredRepositories(this._repositories!);
|
||||||
const repository = repositories.find((repo) => {
|
const repository = repositories.find((repo) => repo.slug === slug);
|
||||||
return repo.slug === slug;
|
|
||||||
});
|
|
||||||
if (!repository) {
|
if (!repository) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const newRepositories = repositories
|
const newRepositories = repositories
|
||||||
.map((repo) => {
|
.map((repo) => repo.source)
|
||||||
return repo.source;
|
.filter((repo) => repo !== repository.source);
|
||||||
})
|
|
||||||
.filter((repo) => {
|
|
||||||
return repo !== repository.source;
|
|
||||||
});
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await setSupervisorOption(this.hass, {
|
await setSupervisorOption(this.hass, {
|
||||||
addons_repositories: newRepositories,
|
addons_repositories: newRepositories,
|
||||||
});
|
});
|
||||||
|
await this._loadData();
|
||||||
const addonsInfo = await fetchHassioAddonsInfo(this.hass);
|
|
||||||
this._repos = addonsInfo.repositories;
|
|
||||||
|
|
||||||
await this._dialogParams!.loadData();
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = extractApiErrorMessage(err);
|
this._error = extractApiErrorMessage(err);
|
||||||
}
|
}
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user