mirror of
https://github.com/emsesp/EMS-ESP32.git
synced 2026-03-24 18:46:33 +03:00
Compare commits
952 Commits
2b2c86ba5a
...
core3
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
c0bea66d27 | ||
|
|
ed7cc078ed | ||
|
|
60b7d6d795 | ||
|
|
947f29cca0 | ||
|
|
d2a13ec0da | ||
|
|
cc39ba409e | ||
|
|
ac9db6256e | ||
|
|
096f628d97 | ||
|
|
bbc2de08a5 | ||
|
|
22312812bb | ||
|
|
df808a2bcf | ||
|
|
d04e7c36f3 | ||
|
|
205d826adb | ||
|
|
3584975acb | ||
|
|
30b9ca4e6c | ||
|
|
7c6ff01ebe | ||
|
|
a54edcaf5b | ||
|
|
e446954844 | ||
|
|
4a2d0d6787 | ||
|
|
9725314135 | ||
|
|
4db8e43648 | ||
|
|
e610f0d57f | ||
|
|
8244af2940 | ||
|
|
cc60062678 | ||
|
|
40f371d23b | ||
|
|
817b791e59 | ||
|
|
25a7aac360 | ||
|
|
37115a174d | ||
|
|
1397f81fd0 | ||
|
|
56365cb403 | ||
|
|
dfd245ee7b | ||
|
|
9c81e4b34d | ||
|
|
67676df131 | ||
|
|
a73b129596 | ||
|
|
4600d886b5 | ||
|
|
e3305ab9db | ||
|
|
0fe45a2405 | ||
|
|
db87213242 | ||
|
|
b0157f288e | ||
|
|
5c3c010d5a | ||
|
|
c804cedd7a | ||
|
|
a9f50d9371 | ||
|
|
65a3226404 | ||
|
|
45690f5418 | ||
|
|
aa30ca99bf | ||
|
|
6836b6197f | ||
|
|
c0ca9d1069 | ||
|
|
5e79e1d57f | ||
|
|
8c732f9f1e | ||
|
|
5e94c2f636 | ||
|
|
69d4163b9d | ||
|
|
b1e974a82c | ||
|
|
34a2b20be8 | ||
|
|
f1fc8d9aae | ||
|
|
b04355e3e1 | ||
|
|
cd3ae5cdf2 | ||
|
|
a261ca23af | ||
|
|
cb96904a5c | ||
|
|
4a2d78f8e1 | ||
|
|
f5af4fb52f | ||
|
|
2037bc3a10 | ||
|
|
64d17d7c65 | ||
|
|
92e2633342 | ||
|
|
96a7ea8a02 | ||
|
|
5c4aaa4510 | ||
|
|
c05e1cb77b | ||
|
|
5879ce4090 | ||
|
|
ac3e5c793c | ||
|
|
64e5d29996 | ||
|
|
b320d8ded2 | ||
|
|
4326fb931b | ||
|
|
ced7051ce7 | ||
|
|
0be1b20996 | ||
|
|
6c55460622 | ||
|
|
421da246ed | ||
|
|
d627404dc2 | ||
|
|
148a721e17 | ||
|
|
f317123c26 | ||
|
|
e4df1887b0 | ||
|
|
34142c3e85 | ||
|
|
6e7f8bdf02 | ||
|
|
3dd9fcfb58 | ||
|
|
35e2954b8b | ||
|
|
59aa63db0f | ||
|
|
7a41a190f8 | ||
|
|
6741232450 | ||
|
|
a811670c5a | ||
|
|
72f08a86cf | ||
|
|
27c471f45f | ||
|
|
e303972d26 | ||
|
|
97bb03d703 | ||
|
|
e9f77c1bde | ||
|
|
81cba6c0a8 | ||
|
|
89029df25e | ||
|
|
3463b6818d | ||
|
|
349843e666 | ||
|
|
96ae3bbbba | ||
|
|
b153364b60 | ||
|
|
ccc40937fb | ||
|
|
909edf394d | ||
|
|
ac8ef646e9 | ||
|
|
3ef279ea10 | ||
|
|
769beeda37 | ||
|
|
f83404c216 | ||
|
|
c239658131 | ||
|
|
be82afd778 | ||
|
|
2156c96368 | ||
|
|
f7f078d82a | ||
|
|
95168cf514 | ||
|
|
6dc601c4a2 | ||
|
|
6b87bbb882 | ||
|
|
abdf2c5037 | ||
|
|
7beec1b80f | ||
|
|
3a0e46f064 | ||
|
|
85cc85a923 | ||
|
|
ca0079c0df | ||
|
|
28b662ad43 | ||
|
|
6bac6bbfeb | ||
|
|
958ec1002b | ||
|
|
438852ecaf | ||
|
|
92d82c0a00 | ||
|
|
7d37267f57 | ||
|
|
5641d53cc3 | ||
|
|
8fc6752290 | ||
|
|
cca6f87500 | ||
|
|
758d76051f | ||
|
|
95f7e66cff | ||
|
|
4e194287c9 | ||
|
|
3545830552 | ||
|
|
074f4c32ed | ||
|
|
b3fec5ed7d | ||
|
|
ffb90b8f9a | ||
|
|
584618043d | ||
|
|
d702c485b7 | ||
|
|
d3561da331 | ||
|
|
0e0aaf37df | ||
|
|
5ec068409f | ||
|
|
8796b6d340 | ||
|
|
bfbb18655d | ||
|
|
9088651e53 | ||
|
|
3e8f379502 | ||
|
|
265c2c4231 | ||
|
|
f671d79280 | ||
|
|
97c89d1d13 | ||
|
|
e0a26a38fa | ||
|
|
038f06e59f | ||
|
|
f4d2bae04f | ||
|
|
d443e275ea | ||
|
|
30d2057e01 | ||
|
|
d952b9aaae | ||
|
|
3402215e8d | ||
|
|
f01031dc26 | ||
|
|
01d4d116b9 | ||
|
|
bc7f82eef1 | ||
|
|
efdb355033 | ||
|
|
87c9fd010f | ||
|
|
930f0e615a | ||
|
|
4dbbdb3290 | ||
|
|
8379b3f5aa | ||
|
|
08124fa4db | ||
|
|
39414db732 | ||
|
|
da57a08005 | ||
|
|
b6b8700c3f | ||
|
|
8ee0789dad | ||
|
|
1eabe86015 | ||
|
|
5fe4b315f3 | ||
|
|
03ec96bf96 | ||
|
|
aa05e37fbb | ||
|
|
959a00c19a | ||
|
|
b42060be3a | ||
|
|
33bb433d7e | ||
|
|
28a5d4ef1a | ||
|
|
b78d47cbd0 | ||
|
|
8a7a1383a7 | ||
|
|
3f5163c1e4 | ||
|
|
fad82c8c68 | ||
|
|
6fc3bf30b6 | ||
|
|
43b3e74c08 | ||
|
|
64c9882d8c | ||
|
|
a690510903 | ||
|
|
c732ec301a | ||
|
|
cc1f16596a | ||
|
|
66c74f85a4 | ||
|
|
db667b9437 | ||
|
|
8799015f59 | ||
|
|
d71b3c64e1 | ||
|
|
6d3083fff4 | ||
|
|
56b2c111b8 | ||
|
|
12ce736580 | ||
|
|
debe90eb8d | ||
|
|
8d318143a4 | ||
|
|
59f90f5d1c | ||
|
|
0efbd0528e | ||
|
|
3218620a0e | ||
|
|
a93921c875 | ||
|
|
fb77b455be | ||
|
|
b64c392c58 | ||
|
|
7bc6cf3910 | ||
|
|
e19e76546e | ||
|
|
b9aaaae90a | ||
|
|
86b395d612 | ||
|
|
6204b9acc7 | ||
|
|
0777f420b0 | ||
|
|
4a6a662aa0 | ||
|
|
66837d399f | ||
|
|
34ff5f12ea | ||
|
|
82d160cabb | ||
|
|
2c85d3829b | ||
|
|
cc258dae16 | ||
|
|
577964befe | ||
|
|
f2500013ab | ||
|
|
98fb6941d2 | ||
|
|
7308b8e73d | ||
|
|
e97cfaf9ee | ||
|
|
fd0734d8d8 | ||
|
|
739f32f045 | ||
|
|
b66b49e812 | ||
|
|
d4b81a2909 | ||
|
|
fb57537e88 | ||
|
|
335b1274cf | ||
|
|
f6d1c87eaf | ||
|
|
5e07e9a11b | ||
|
|
dd0ea5df0e | ||
|
|
2b7f592957 | ||
|
|
db7b5df85d | ||
|
|
1c2534ed8f | ||
|
|
90535d7b94 | ||
|
|
0d3a8fc719 | ||
|
|
d624b9eac9 | ||
|
|
068cbf757c | ||
|
|
738d6f0b0f | ||
|
|
d9551bc4c3 | ||
|
|
722659325a | ||
|
|
927e7c80f4 | ||
|
|
39919b4ad8 | ||
|
|
b5defa552e | ||
|
|
e8fbbe5a1c | ||
|
|
949172128f | ||
|
|
35a8db4581 | ||
|
|
3a74abb4db | ||
|
|
bcc7687b1b | ||
|
|
13aa544214 | ||
|
|
696141721a | ||
|
|
4781eea665 | ||
|
|
2fd6bed485 | ||
|
|
db40d1d381 | ||
|
|
33bde8b407 | ||
|
|
bf5990a992 | ||
|
|
cff4bd0a71 | ||
|
|
28ee0834d8 | ||
|
|
9be1cb1d3e | ||
|
|
81d46fede2 | ||
|
|
664a8e9f5f | ||
|
|
def7501c62 | ||
|
|
1cc4dc52d4 | ||
|
|
978c738f27 | ||
|
|
feeb8500ac | ||
|
|
18a55d4622 | ||
|
|
29ea67f438 | ||
|
|
a3df77171b | ||
|
|
aa6f5c50b2 | ||
|
|
53a43ca147 | ||
|
|
da76fe3871 | ||
|
|
84af132e2c | ||
|
|
712a8537c9 | ||
|
|
bb22386f7f | ||
|
|
89dfe11ee3 | ||
|
|
be1e08af9c | ||
|
|
68ebcdded4 | ||
|
|
4afe041880 | ||
|
|
8b690d23da | ||
|
|
62c7fb671b | ||
|
|
5a82064a88 | ||
|
|
4ae4000944 | ||
|
|
616c73f658 | ||
|
|
b698485814 | ||
|
|
b4036bf8cd | ||
|
|
4b457d6cdb | ||
|
|
425b44e334 | ||
|
|
41bf293db3 | ||
|
|
af349edd54 | ||
|
|
5b303bd58a | ||
|
|
b992f90fe2 | ||
|
|
92c34dddba | ||
|
|
1475fc094d | ||
|
|
48f7b48216 | ||
|
|
cd054b293a | ||
|
|
ea6b7c0be0 | ||
|
|
a49a5537d3 | ||
|
|
c407ad04bf | ||
|
|
480e0951b8 | ||
|
|
a06c5bb297 | ||
|
|
77a54792dd | ||
|
|
d0d49397ca | ||
|
|
b51abeabac | ||
|
|
f0e4f17ab8 | ||
|
|
205da33fe5 | ||
|
|
9aa78111be | ||
|
|
c3f93d4aae | ||
|
|
5a5c0d7179 | ||
|
|
ba57942b7d | ||
|
|
c782deb581 | ||
|
|
566edfcd7b | ||
|
|
c3cf38c330 | ||
|
|
4953a41330 | ||
|
|
7ff4ed640d | ||
|
|
898a13fcb5 | ||
|
|
3fdc370466 | ||
|
|
39055ad0d2 | ||
|
|
21e73c973a | ||
|
|
4d03976032 | ||
|
|
eb7587270f | ||
|
|
3ea88a2be0 | ||
|
|
55778ba0b5 | ||
|
|
5d3694abd3 | ||
|
|
21d9ba0182 | ||
|
|
27848feddd | ||
|
|
ce261eeb65 | ||
|
|
8de3ae5468 | ||
|
|
367d27d48f | ||
|
|
1a9b6ab2a5 | ||
|
|
323b8ee67d | ||
|
|
5b95e1d41f | ||
|
|
a272d8e253 | ||
|
|
79285ca12e | ||
|
|
f90f676faf | ||
|
|
3224d8823d | ||
|
|
1a03b98670 | ||
|
|
80f32bfeb4 | ||
|
|
1b4693b981 | ||
|
|
535b760dd7 | ||
|
|
14775f6503 | ||
|
|
a856d249c9 | ||
|
|
484b547df5 | ||
|
|
d73ca2c890 | ||
|
|
6211bd8c69 | ||
|
|
e638a471d1 | ||
|
|
42ee21e883 | ||
|
|
263af58dc0 | ||
|
|
6727c0655a | ||
|
|
cba249938a | ||
|
|
9fbed47617 | ||
|
|
b5dd722888 | ||
|
|
11bef52568 | ||
|
|
d22a369333 | ||
|
|
dfe95296d9 | ||
|
|
8d39893e5e | ||
|
|
b8b8a501e1 | ||
|
|
2a36f378b4 | ||
|
|
6746df37a1 | ||
|
|
364f66b7d4 | ||
|
|
84bbd93216 | ||
|
|
85ef8d7d50 | ||
|
|
2b6606d8ad | ||
|
|
4772a61e7c | ||
|
|
d30375f3c4 | ||
|
|
8c831ac0e9 | ||
|
|
1ede7028a5 | ||
|
|
5b8dd0a693 | ||
|
|
cc041510be | ||
|
|
05baec85b7 | ||
|
|
fb698fd029 | ||
|
|
bbfec136e8 | ||
|
|
36271a2c24 | ||
|
|
bbe1f133dc | ||
|
|
d7b0614556 | ||
|
|
7e9f27a613 | ||
|
|
3bdc97d4d5 | ||
|
|
8a7511a941 | ||
|
|
a9511e6a29 | ||
|
|
ac37ead419 | ||
|
|
39fcda59da | ||
|
|
9c44e104bb | ||
|
|
18f8db7942 | ||
|
|
71281bc82d | ||
|
|
0557def0b6 | ||
|
|
1a3f7fbbee | ||
|
|
7bed8bf84e | ||
|
|
790371a9e2 | ||
|
|
537cf19e97 | ||
|
|
35ad43b7b3 | ||
|
|
3d70e8c1e6 | ||
|
|
7bb30d37ce | ||
|
|
0267f00b48 | ||
|
|
5a7cec91c5 | ||
|
|
642bf63abc | ||
|
|
6f3197f482 | ||
|
|
5c88968879 | ||
|
|
c4a43183b3 | ||
|
|
94b583d7f3 | ||
|
|
37012e55e3 | ||
|
|
ea4d613d12 | ||
|
|
0e6108b5a9 | ||
|
|
fdbaf7509f | ||
|
|
74182031ae | ||
|
|
6c80a34578 | ||
|
|
09f1c13d28 | ||
|
|
5668fe13ae | ||
|
|
78a02b6d85 | ||
|
|
ccab932e8d | ||
|
|
eaea1f383b | ||
|
|
de8309de4a | ||
|
|
bc3269037f | ||
|
|
31131427b8 | ||
|
|
9957bff62b | ||
|
|
f1841347a7 | ||
|
|
1b8b72c443 | ||
|
|
b4affbff6d | ||
|
|
c8b4a38bb6 | ||
|
|
eec373e2ae | ||
|
|
48b261317d | ||
|
|
e6beb01075 | ||
|
|
ecc6e9286a | ||
|
|
179351cb6b | ||
|
|
778fe43012 | ||
|
|
d84d52df4b | ||
|
|
11d4109915 | ||
|
|
0eddbac150 | ||
|
|
99afeb221a | ||
|
|
39d18b78a1 | ||
|
|
4ebe8cc0cc | ||
|
|
6dabfb7fe2 | ||
|
|
611b1d9aca | ||
|
|
2821f8e750 | ||
|
|
1cccd8dc2c | ||
|
|
4d13982594 | ||
|
|
10c63640c0 | ||
|
|
14ad1239db | ||
|
|
2a16cb6e64 | ||
|
|
89fa5947bd | ||
|
|
a81e56e3bf | ||
|
|
90ad2dde54 | ||
|
|
9c243cbe8d | ||
|
|
c1b444541f | ||
|
|
8527b16e9d | ||
|
|
a728420010 | ||
|
|
378d9e8634 | ||
|
|
1973081529 | ||
|
|
45ae6d802c | ||
|
|
c4297e2996 | ||
|
|
310afe3ab8 | ||
|
|
3215f36530 | ||
|
|
55621e12d9 | ||
|
|
458dc516f4 | ||
|
|
f2ae84bd22 | ||
|
|
f093df1cb9 | ||
|
|
05f15f7876 | ||
|
|
12f4a74094 | ||
|
|
15d895fd0a | ||
|
|
1890948924 | ||
|
|
fec246127f | ||
|
|
ecab30d4ac | ||
|
|
f8cc688241 | ||
|
|
f51b3528d5 | ||
|
|
42c94a1017 | ||
|
|
acccb56f07 | ||
|
|
3a2d3ac985 | ||
|
|
0ab18e6e08 | ||
|
|
44db5991e7 | ||
|
|
15a6c50326 | ||
|
|
911aa40ca1 | ||
|
|
2b679daabc | ||
|
|
71b956e613 | ||
|
|
79089a93bc | ||
|
|
da3ac1794e | ||
|
|
bc870b2aa2 | ||
|
|
6bc40ce2e1 | ||
|
|
84544979fa | ||
|
|
2446e4d1fd | ||
|
|
4a15b39945 | ||
|
|
df080bbad9 | ||
|
|
8632af8820 | ||
|
|
27047c0f39 | ||
|
|
95de3e339d | ||
|
|
99f44aece5 | ||
|
|
4e589aecbf | ||
|
|
ab013554bd | ||
|
|
5c966e291b | ||
|
|
068744b681 | ||
|
|
8c794baa7b | ||
|
|
f6a23723f8 | ||
|
|
4b3cc2e3ec | ||
|
|
4ee743d309 | ||
|
|
d3b4bab40a | ||
|
|
c71b54fb5b | ||
|
|
ecbdf01bdf | ||
|
|
c2e8a6c73d | ||
|
|
29037d19fb | ||
|
|
67d242d210 | ||
|
|
91a67def99 | ||
|
|
c8ab89ef37 | ||
|
|
17a9b7eb0a | ||
|
|
cdb592d744 | ||
|
|
af19941a07 | ||
|
|
00dce83096 | ||
|
|
597e60d6f1 | ||
|
|
4e4311cac0 | ||
|
|
c11402195f | ||
|
|
31ec15811f | ||
|
|
739ac7b42e | ||
|
|
ad3049ab0a | ||
|
|
d96c3f3ed7 | ||
|
|
dcfd0d5b11 | ||
|
|
b05712cf83 | ||
|
|
df15485d7c | ||
|
|
bb15fcdc46 | ||
|
|
e34291fbd5 | ||
|
|
55b8c2d04c | ||
|
|
8bd1cd03b4 | ||
|
|
bafb7c35fb | ||
|
|
9425558ff7 | ||
|
|
f20aad3813 | ||
|
|
7a683d3637 | ||
|
|
ac982cbb15 | ||
|
|
9889b1b5c4 | ||
|
|
7cb647e5f1 | ||
|
|
515b75160c | ||
|
|
a365dc7519 | ||
|
|
764520714b | ||
|
|
43e087ae91 | ||
|
|
bffccc585a | ||
|
|
a01f10b042 | ||
|
|
26a5f98aae | ||
|
|
67280546af | ||
|
|
64058b0f61 | ||
|
|
d7b5c81b0e | ||
|
|
1d03056784 | ||
|
|
dd0ab8f962 | ||
|
|
02e8dba971 | ||
|
|
59878fb190 | ||
|
|
9ff0f83af9 | ||
|
|
e6f825371e | ||
|
|
45f3f23033 | ||
|
|
ffd27db208 | ||
|
|
a452d6131b | ||
|
|
0bc478f9d3 | ||
|
|
03ef981765 | ||
|
|
9ca9f25fd3 | ||
|
|
41122dddb2 | ||
|
|
1e0c94d007 | ||
|
|
3e42a7fb4c | ||
|
|
a8fcc1fb44 | ||
|
|
e43416019d | ||
|
|
9f467ecec1 | ||
|
|
273d87dbf1 | ||
|
|
befd21f8cb | ||
|
|
882e3bc1cd | ||
|
|
15e05c4abc | ||
|
|
748a2f5fcf | ||
|
|
37ba42faf8 | ||
|
|
19e343e517 | ||
|
|
8f39129bf8 | ||
|
|
9c3521caf2 | ||
|
|
40fc0fd2f9 | ||
|
|
ff8566498f | ||
|
|
dd06882860 | ||
|
|
fb2294c945 | ||
|
|
26ea8320ce | ||
|
|
0f4963d91e | ||
|
|
91020abc90 | ||
|
|
64906f3ea0 | ||
|
|
28f85b4c5a | ||
|
|
b44a0d6813 | ||
|
|
8af7cde2d6 | ||
|
|
3fcd656bb6 | ||
|
|
76c827257e | ||
|
|
8ca9f7ee30 | ||
|
|
da7a06646a | ||
|
|
0a36f1df7a | ||
|
|
80e5d30781 | ||
|
|
9c4beba3b1 | ||
|
|
0cf932f57e | ||
|
|
5cb9f3b014 | ||
|
|
3eb581142a | ||
|
|
d6c460e7fd | ||
|
|
a2baa50530 | ||
|
|
6569b8c038 | ||
|
|
48b4bf02a3 | ||
|
|
693054a92a | ||
|
|
1472e53410 | ||
|
|
5ed4970d62 | ||
|
|
056cf3cbd6 | ||
|
|
2bcd548747 | ||
|
|
9edcf47073 | ||
|
|
084d90e714 | ||
|
|
a738cc36dd | ||
|
|
9b6f9aeda3 | ||
|
|
8ea8c1821d | ||
|
|
53a1a8826e | ||
|
|
237551d9f6 | ||
|
|
f7c7bc65f2 | ||
|
|
badbd9c6fe | ||
|
|
cc7ac5b911 | ||
|
|
5957300c4e | ||
|
|
2e343ce0c3 | ||
|
|
3113a4be2b | ||
|
|
7d9cb2932f | ||
|
|
026c2caea0 | ||
|
|
eb058b688d | ||
|
|
2a7a0ce3f6 | ||
|
|
a330110eef | ||
|
|
feed534be5 | ||
|
|
2375633bca | ||
|
|
7018139289 | ||
|
|
5f67934f26 | ||
|
|
574cc8ad33 | ||
|
|
658c613c4e | ||
|
|
c2ce54c2a7 | ||
|
|
58da0d9778 | ||
|
|
8b90036c11 | ||
|
|
65678ea739 | ||
|
|
34ddc9b7ff | ||
|
|
97f9914d33 | ||
|
|
8b64851f6f | ||
|
|
c00f50238e | ||
|
|
25ea57e02f | ||
|
|
a951afe205 | ||
|
|
44c7954ce7 | ||
|
|
ef315b6dde | ||
|
|
935a9dcbb7 | ||
|
|
2bda432d70 | ||
|
|
d567ea3cf0 | ||
|
|
7b5e386595 | ||
|
|
1c65a7caba | ||
|
|
12ebacf1a7 | ||
|
|
f2c176111f | ||
|
|
cc242e5eba | ||
|
|
4888808ad0 | ||
|
|
cd1b5e1d57 | ||
|
|
9661c9a0eb | ||
|
|
f6ccf6da44 | ||
|
|
b691488240 | ||
|
|
eb71996e6a | ||
|
|
b9566ae1d6 | ||
|
|
8016fc4287 | ||
|
|
c95b43ea69 | ||
|
|
d767a503cd | ||
|
|
221131f9d3 | ||
|
|
f307cccabb | ||
|
|
2f21d0d94c | ||
|
|
132d1292ce | ||
|
|
df9a10cb53 | ||
|
|
95d564901b | ||
|
|
6424d9b1c0 | ||
|
|
38a3d20acf | ||
|
|
89b117bbc2 | ||
|
|
eeba7a3a6b | ||
|
|
23a660aabb | ||
|
|
c9bddba446 | ||
|
|
3a508a3ec4 | ||
|
|
6bf33f6447 | ||
|
|
a02054ceb6 | ||
|
|
8422521975 | ||
|
|
fbfacc5ed5 | ||
|
|
b9d96620a4 | ||
|
|
8c61735579 | ||
|
|
56e8ccdfc4 | ||
|
|
5454e7dd16 | ||
|
|
901a27140c | ||
|
|
37db2b9504 | ||
|
|
5e7768f912 | ||
|
|
80dd16740d | ||
|
|
24421a0224 | ||
|
|
26d26cf088 | ||
|
|
615c5e8439 | ||
|
|
090387ef37 | ||
|
|
25ea7d8b0c | ||
|
|
68f067f2c4 | ||
|
|
e20fa5ab39 | ||
|
|
bee307a91d | ||
|
|
f85226ce55 | ||
|
|
f112e6f6cc | ||
|
|
5df82b7e2c | ||
|
|
ea75a34c82 | ||
|
|
07d0de0151 | ||
|
|
0a75dd7e3c | ||
|
|
88afd3f453 | ||
|
|
9669a044ba | ||
|
|
e31ebab12b | ||
|
|
e68a3a0d3a | ||
|
|
4ae09c8766 | ||
|
|
a693e96248 | ||
|
|
60b0de79b3 | ||
|
|
86277396f7 | ||
|
|
0b65d55601 | ||
|
|
23782ec773 | ||
|
|
0b33497842 | ||
|
|
6cd4bcd41c | ||
|
|
8d8db3ba85 | ||
|
|
d9c2066035 | ||
|
|
af5cbf045d | ||
|
|
99d67cdd42 | ||
|
|
a74910ddf6 | ||
|
|
d9678d04dd | ||
|
|
c7acf89d84 | ||
|
|
24f1fe13cc | ||
|
|
11fcd8bcfe | ||
|
|
bcde5bad63 | ||
|
|
d2a8fbaf1e | ||
|
|
f9d2b18959 | ||
|
|
bdba2ae6e4 | ||
|
|
f068ed97f1 | ||
|
|
7ab2e782ac | ||
|
|
88a7d12306 | ||
|
|
dffe8b2648 | ||
|
|
daa98b106e | ||
|
|
83796341c5 | ||
|
|
ea24cd8a0f | ||
|
|
0e7ff32d02 | ||
|
|
31706bfc21 | ||
|
|
993e2fdc22 | ||
|
|
0d05cf489b | ||
|
|
0961b814d5 | ||
|
|
999c407f12 | ||
|
|
7c137d71ca | ||
|
|
ad26768d6d | ||
|
|
5ed7d60ae8 | ||
|
|
a67379fcb2 | ||
|
|
cccae3fbcc | ||
|
|
10d85c6547 | ||
|
|
21af1adefe | ||
|
|
fe02258a00 | ||
|
|
3d8ec8e295 | ||
|
|
d5b496aa67 | ||
|
|
8458cfc468 | ||
|
|
f10d2ef48d | ||
|
|
2ee433bb89 | ||
|
|
237cf91f5a | ||
|
|
8db7e4a5b9 | ||
|
|
a3703514b1 | ||
|
|
db8ed34dc4 | ||
|
|
21b89cf84d | ||
|
|
fc0f29337b | ||
|
|
84b9e6ae18 | ||
|
|
cbabaa7d91 | ||
|
|
4cfe704242 | ||
|
|
25cae61cc1 | ||
|
|
d96b3e6d05 | ||
|
|
1f261476d5 | ||
|
|
3171fc7375 | ||
|
|
ba843c1589 | ||
|
|
f7f01ea875 | ||
|
|
8ea1dfc7d2 | ||
|
|
9616a113b0 | ||
|
|
4aaf6e95cf | ||
|
|
edfdfb1016 | ||
|
|
3067149357 | ||
|
|
d5548c8bdc | ||
|
|
cf66dc99b4 | ||
|
|
891619fa26 | ||
|
|
1f6462be38 | ||
|
|
e74dc1fd78 | ||
|
|
f59768ce17 | ||
|
|
22d015615d | ||
|
|
0ef0ca8518 | ||
|
|
1917c5d7cb | ||
|
|
ff0fe593d3 | ||
|
|
12e8d64ec2 | ||
|
|
c0c13eb687 | ||
|
|
91b78f9a23 | ||
|
|
32474d10ce | ||
|
|
7c3de25c20 | ||
|
|
f75b7b1a59 | ||
|
|
a3e01b8a3b | ||
|
|
6c9a9b8632 | ||
|
|
3fba75868f | ||
|
|
8dee390d75 | ||
|
|
dc838639b2 | ||
|
|
3fd05c8eb7 | ||
|
|
5f0df140b0 | ||
|
|
b98cbd3ec5 | ||
|
|
18d67d088e | ||
|
|
0bf60394fe | ||
|
|
cec5ffd547 | ||
|
|
026ea4450e | ||
|
|
9a1dd5bb98 | ||
|
|
fbc42fbb15 | ||
|
|
5613cde00f | ||
|
|
d65d6f49cd | ||
|
|
c9bc18cf4b | ||
|
|
9179127bce | ||
|
|
16b6bef393 | ||
|
|
52159ac596 | ||
|
|
0dc3fd43e9 | ||
|
|
b0d490036f | ||
|
|
c28b098c65 | ||
|
|
4a6ccce09a | ||
|
|
5053ad08dd | ||
|
|
72b4809ed8 | ||
|
|
6799fe5189 | ||
|
|
12635ff4a5 | ||
|
|
5908fd9d9c | ||
|
|
5c07a2c0cc | ||
|
|
f5048abae7 | ||
|
|
76acffdb2e | ||
|
|
1ac96aa02e | ||
|
|
3386ac7f8a | ||
|
|
1e4c157c28 | ||
|
|
a1c5297eef | ||
|
|
cda04bef26 | ||
|
|
1edf60b617 | ||
|
|
32cf4dd6c9 | ||
|
|
f0caeb089d | ||
|
|
23f1e7569c | ||
|
|
5087fdb5d6 | ||
|
|
b654a42229 | ||
|
|
d312c8e592 | ||
|
|
319787bae3 | ||
|
|
d124b04a2c | ||
|
|
4ef8a3a163 | ||
|
|
41b5cdddf2 | ||
|
|
e10453b2fd | ||
|
|
d2f665ab70 | ||
|
|
4083478b65 | ||
|
|
01136a19a5 | ||
|
|
0b2df96461 | ||
|
|
f66c3ff322 | ||
|
|
5d8fe89e5a | ||
|
|
22eb5436ab | ||
|
|
17bdd87576 | ||
|
|
8f7c0a1d97 | ||
|
|
3b453b18bc | ||
|
|
a1fc5bf54b | ||
|
|
0cb413a579 | ||
|
|
d326d345f4 | ||
|
|
1de647e8a5 | ||
|
|
c562b88b32 | ||
|
|
f2e38330ea | ||
|
|
f2dca2a90a | ||
|
|
9f1cd04d45 | ||
|
|
fe67f3a982 | ||
|
|
2fd6e9ebf5 | ||
|
|
0f1195de82 | ||
|
|
267042f74f | ||
|
|
ea9783dbfc | ||
|
|
67655c4b06 | ||
|
|
2970aa5ba3 | ||
|
|
99a3ffcf17 | ||
|
|
9bc5bde3c2 | ||
|
|
0edb844225 | ||
|
|
e3feb8f11e | ||
|
|
6b7534b7fb | ||
|
|
ca1506de8b | ||
|
|
1cb535dea3 | ||
|
|
5d32b6d383 | ||
|
|
c203b8e6d2 | ||
|
|
e6dbe020c1 | ||
|
|
63cf1603b0 | ||
|
|
c184a0af10 | ||
|
|
8a4e6d5ed5 | ||
|
|
aadb67fa79 | ||
|
|
4949471518 | ||
|
|
9504723ef2 | ||
|
|
3abfb7bb9c | ||
|
|
3ef6bf8681 | ||
|
|
1647a10b1b | ||
|
|
55b893362c | ||
|
|
48f1928327 | ||
|
|
dc793f145d | ||
|
|
77e0b7d89c | ||
|
|
cec19860bc | ||
|
|
898acb90d0 | ||
|
|
a074ac732d | ||
|
|
c71526b95c | ||
|
|
f035e31dcf | ||
|
|
35192b9dde | ||
|
|
b5ac637231 | ||
|
|
4e0ef8b0e3 | ||
|
|
c32ee4dfb5 | ||
|
|
accfeab6fa | ||
|
|
d98f3cc8c5 | ||
|
|
342f238983 | ||
|
|
73f8ea0fc5 | ||
|
|
f2bd1ff575 | ||
|
|
b3ec23d6bd | ||
|
|
1fc7fa4720 | ||
|
|
58ae058465 | ||
|
|
7ece395d1b | ||
|
|
4bb876031e | ||
|
|
e4f129db04 | ||
|
|
90038e08dc | ||
|
|
4c30e930cf | ||
|
|
dd69e02f6b | ||
|
|
f2ccf1953d | ||
|
|
0e133840c9 | ||
|
|
e12b277472 | ||
|
|
dc7b1809c1 | ||
|
|
8c2146ff55 | ||
|
|
882d412409 | ||
|
|
3f88a9469c | ||
|
|
3ac7d43a81 | ||
|
|
d12446b6d9 | ||
|
|
865fb3a967 | ||
|
|
48538222b2 | ||
|
|
8d44f61517 | ||
|
|
0e4f6f4209 | ||
|
|
a2823563bf | ||
|
|
353cdb324d | ||
|
|
216f799db1 | ||
|
|
4d6f080263 | ||
|
|
b9d124618c | ||
|
|
ab5d9e8d36 | ||
|
|
4cdeecd952 | ||
|
|
59b3933cb6 | ||
|
|
179ddcb348 | ||
|
|
efa2c8fc4b | ||
|
|
c958c7d61a | ||
|
|
35fca9c450 | ||
|
|
61962fbc07 | ||
|
|
39e724befe | ||
|
|
43bb77b095 | ||
|
|
28e1e46586 | ||
|
|
2f5b879652 | ||
|
|
16930fe8ca | ||
|
|
1db1b6e524 | ||
|
|
43eba7a010 | ||
|
|
49278bdea4 | ||
|
|
2405e11af2 | ||
|
|
23b6894484 | ||
|
|
a837c9398c | ||
|
|
ea484e15f9 | ||
|
|
cce99a8b1d | ||
|
|
f29faafd78 | ||
|
|
c3eafbcd85 | ||
|
|
65e18ab4e2 | ||
|
|
8bb16ed3a7 | ||
|
|
e685284f72 | ||
|
|
86d2805642 | ||
|
|
29b98a15a4 | ||
|
|
ba9df92b12 | ||
|
|
0eac1c9bf9 | ||
|
|
515feb9f9e | ||
|
|
0b84b79e1d | ||
|
|
8fd129f4fe | ||
|
|
ba334930fe | ||
|
|
982d64ddca | ||
|
|
3577300361 | ||
|
|
0478e0ff7c | ||
|
|
c2cfd0a1b0 | ||
|
|
637ba30df1 | ||
|
|
fa8421b297 | ||
|
|
61c23a57d9 | ||
|
|
43f61fd2df | ||
|
|
85eed64fe9 | ||
|
|
43634a4312 | ||
|
|
9757db4438 | ||
|
|
17a2ba7f1a | ||
|
|
03c7417888 | ||
|
|
99b43b0379 | ||
|
|
6fb8fbba18 | ||
|
|
09c750e622 | ||
|
|
b4f174f2f4 |
27
.devcontainer/devcontainer.json
Normal file
27
.devcontainer/devcontainer.json
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
{
|
||||||
|
"name": "EMS-ESP Devcontainer",
|
||||||
|
"image": "mcr.microsoft.com/devcontainers/typescript-node:1-22-bookworm",
|
||||||
|
"features": {
|
||||||
|
"ghcr.io/devcontainers/features/node:1": {},
|
||||||
|
"ghcr.io/devcontainers-extra/features/pnpm:2": {},
|
||||||
|
"ghcr.io/devcontainers/features/python:1": {},
|
||||||
|
"ghcr.io/shyim/devcontainers-features/bun:0": {}
|
||||||
|
},
|
||||||
|
|
||||||
|
"forwardPorts": [
|
||||||
|
3000,
|
||||||
|
3080
|
||||||
|
],
|
||||||
|
|
||||||
|
// Use 'postCreateCommand' to run commands after the container is created.
|
||||||
|
"postCreateCommand": "cd mock-api && pnpm install && cd .. && cd interface && pnpm install",
|
||||||
|
|
||||||
|
// Configure tool-specific properties.
|
||||||
|
"customizations": {
|
||||||
|
"vscode": {
|
||||||
|
"extensions": [
|
||||||
|
"platformio.platformio-ide"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
4
.github/ISSUE_TEMPLATE/bug_report.md
vendored
4
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@@ -21,8 +21,8 @@ _Make sure your have performed every step and checked the applicable boxes befor
|
|||||||
|
|
||||||
- [ ] Searched the issue in [issues](https://github.com/emsesp/EMS-ESP32/issues)
|
- [ ] Searched the issue in [issues](https://github.com/emsesp/EMS-ESP32/issues)
|
||||||
- [ ] Searched the issue in [discussions](https://github.com/emsesp/EMS-ESP32/discussions)
|
- [ ] Searched the issue in [discussions](https://github.com/emsesp/EMS-ESP32/discussions)
|
||||||
- [ ] Searched the issue in the [docs](https://docs.emsesp.org/Troubleshooting/)
|
- [ ] Searched the issue in the [docs](https://emsesp.org/Troubleshooting/)
|
||||||
- [ ] Searched the issue in the [chat](https://discord.gg/3J3GgnzpyT)
|
- [ ] Searched the issue in the [chat](https://discord.gg/GP9DPSgeJq)
|
||||||
- [ ] Provide the System information in the area below, taken from `http://<IP>/api/system`
|
- [ ] Provide the System information in the area below, taken from `http://<IP>/api/system`
|
||||||
|
|
||||||
```json
|
```json
|
||||||
|
|||||||
4
.github/ISSUE_TEMPLATE/config.yml
vendored
4
.github/ISSUE_TEMPLATE/config.yml
vendored
@@ -1,11 +1,11 @@
|
|||||||
blank_issues_enabled: false
|
blank_issues_enabled: false
|
||||||
contact_links:
|
contact_links:
|
||||||
- name: EMS-ESP Docs
|
- name: EMS-ESP Docs
|
||||||
url: https://docs.emsesp.org
|
url: https://emsesp.org
|
||||||
about: All the information related to EMS-ESP.
|
about: All the information related to EMS-ESP.
|
||||||
- name: EMS-ESP Discussions and Support
|
- name: EMS-ESP Discussions and Support
|
||||||
url: https://github.com/emsesp/EMS-ESP32/discussions
|
url: https://github.com/emsesp/EMS-ESP32/discussions
|
||||||
about: EMS-ESP usage Questions, Feature Requests and Projects.
|
about: EMS-ESP usage Questions, Feature Requests and Projects.
|
||||||
- name: EMS-ESP Users Chat
|
- name: EMS-ESP Users Chat
|
||||||
url: https://discord.gg/3J3GgnzpyT
|
url: https://discord.gg/GP9DPSgeJq
|
||||||
about: Chat for feedback, questions and troubleshooting.
|
about: Chat for feedback, questions and troubleshooting.
|
||||||
|
|||||||
36
.github/workflows/dev_release.yml
vendored
36
.github/workflows/dev_release.yml
vendored
@@ -18,17 +18,17 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
|
|
||||||
- name: Install python 3.13
|
- name: Install python 3.13
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v6
|
||||||
with:
|
with:
|
||||||
python-version: '3.13'
|
python-version: '3.13'
|
||||||
|
|
||||||
- name: Install Node.js 22
|
- name: Install Node.js 24
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version: 22
|
node-version: 24
|
||||||
|
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
- name: Enable Corepack
|
- name: Enable Corepack
|
||||||
run: corepack enable pnpm
|
run: corepack enable pnpm
|
||||||
@@ -45,20 +45,26 @@ jobs:
|
|||||||
pip install -U platformio
|
pip install -U platformio
|
||||||
python -m pip install intelhex
|
python -m pip install intelhex
|
||||||
|
|
||||||
- name: Build the WebUI
|
- name: Build webUI
|
||||||
run: |
|
run: |
|
||||||
cd interface
|
platformio run -e build_webUI
|
||||||
pnpm install
|
|
||||||
pnpm typesafe-i18n --no-watch
|
|
||||||
sed -i "s/= 'pl'/= 'en'/" ./src/i18n/i18n-util.ts
|
|
||||||
pnpm build
|
|
||||||
pnpm webUI
|
|
||||||
|
|
||||||
- name: Build all PIO target environments
|
- name: Build modbus
|
||||||
|
run: |
|
||||||
|
platformio run -e build_modbus
|
||||||
|
|
||||||
|
- name: Build standalone
|
||||||
|
run: |
|
||||||
|
platformio run -e build_standalone
|
||||||
|
|
||||||
|
- name: Build all PIO target environments, from default_envs
|
||||||
run: |
|
run: |
|
||||||
platformio run
|
platformio run
|
||||||
env:
|
|
||||||
NO_BUILD_WEBUI: true
|
- name: Commit the generated files
|
||||||
|
uses: stefanzweifel/git-auto-commit-action@v5
|
||||||
|
with:
|
||||||
|
commit_message: "chore: update generated files for v${{steps.build_info.outputs.VERSION}}"
|
||||||
|
|
||||||
- name: Create GitHub Release
|
- name: Create GitHub Release
|
||||||
id: 'automatic_releases'
|
id: 'automatic_releases'
|
||||||
|
|||||||
@@ -7,9 +7,11 @@ on:
|
|||||||
jobs:
|
jobs:
|
||||||
github-releases-to-discord:
|
github-releases-to-discord:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
- name: GitHub Releases To Discord
|
- name: GitHub Releases To Discord
|
||||||
uses: SethCohen/github-releases-to-discord@v1.13.1
|
uses: SethCohen/github-releases-to-discord@v1.13.1
|
||||||
|
|||||||
23
.github/workflows/pr_check.yml
vendored
23
.github/workflows/pr_check.yml
vendored
@@ -1,18 +1,13 @@
|
|||||||
name: 'Pre-check on PR'
|
name: 'Pre-check on PR'
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
pull_request:
|
pull_request:
|
||||||
branches: dev
|
branches: dev
|
||||||
paths:
|
paths:
|
||||||
- '**.c'
|
- 'src/**'
|
||||||
- '**.cpp'
|
|
||||||
- '**.h'
|
|
||||||
- '**.hpp'
|
|
||||||
- '**.json'
|
|
||||||
- '**.py'
|
|
||||||
- '**.md'
|
|
||||||
- '.github/workflows/pr_check.yml'
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
pre-release:
|
pre-release:
|
||||||
@@ -20,18 +15,18 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
- name: Install python 3.11
|
- name: Install python 3.13
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v6
|
||||||
with:
|
with:
|
||||||
python-version: '3.11'
|
python-version: '3.13'
|
||||||
|
|
||||||
- name: Install PlatformIO
|
- name: Install PlatformIO
|
||||||
run: |
|
run: |
|
||||||
pip install wheel
|
pip install wheel
|
||||||
pip install -U platformio
|
pip install -U platformio
|
||||||
|
|
||||||
- name: Build native
|
- name: Run unit tests
|
||||||
run: |
|
run: |
|
||||||
platformio run -e native
|
platformio run -e native-test -t exec
|
||||||
|
|||||||
8
.github/workflows/sonar_check.yml
vendored
8
.github/workflows/sonar_check.yml
vendored
@@ -1,12 +1,14 @@
|
|||||||
# see https://github.com/marketplace/actions/sonarcloud-scan-for-c-and-c#usage
|
# see https://github.com/marketplace/actions/sonarcloud-scan-for-c-and-c#usage
|
||||||
name: Sonar Check
|
name: Sonar Check
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- dev
|
- dev
|
||||||
# pull_request:
|
paths:
|
||||||
# types: [opened, synchronize, reopened]
|
- 'src/**'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
@@ -17,7 +19,7 @@ jobs:
|
|||||||
BUILD_WRAPPER_OUT_DIR: bw-output
|
BUILD_WRAPPER_OUT_DIR: bw-output
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v5
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: Install Build Wrapper
|
- name: Install Build Wrapper
|
||||||
|
|||||||
31
.github/workflows/stable_release.yml
vendored
31
.github/workflows/stable_release.yml
vendored
@@ -16,17 +16,17 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
|
|
||||||
- name: Install python 3.13
|
- name: Install python 3.13
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v6
|
||||||
with:
|
with:
|
||||||
python-version: '3.13'
|
python-version: '3.13'
|
||||||
|
|
||||||
- name: Install Node.js 22
|
- name: Install Node.js 24
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version: 22
|
node-version: 24
|
||||||
|
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
- name: Enable Corepack
|
- name: Enable Corepack
|
||||||
run: corepack enable pnpm
|
run: corepack enable pnpm
|
||||||
@@ -37,20 +37,21 @@ jobs:
|
|||||||
pip install -U platformio
|
pip install -U platformio
|
||||||
python -m pip install intelhex
|
python -m pip install intelhex
|
||||||
|
|
||||||
- name: Build the WebUI
|
- name: Build webUI
|
||||||
run: |
|
run: |
|
||||||
cd interface
|
platformio run -e build_webUI
|
||||||
pnpm install
|
|
||||||
pnpm typesafe-i18n --no-watch
|
|
||||||
sed -i "s/= 'pl'/= 'en'/" ./src/i18n/i18n-util.ts
|
|
||||||
pnpm build
|
|
||||||
pnpm webUI
|
|
||||||
|
|
||||||
- name: Build all PIO target environments
|
- name: Build modbus
|
||||||
|
run: |
|
||||||
|
platformio run -e build_modbus
|
||||||
|
|
||||||
|
- name: Build standalone
|
||||||
|
run: |
|
||||||
|
platformio run -e build_standalone
|
||||||
|
|
||||||
|
- name: Build all PIO target environments, from default_envs
|
||||||
run: |
|
run: |
|
||||||
platformio run
|
platformio run
|
||||||
env:
|
|
||||||
NO_BUILD_WEBUI: true
|
|
||||||
|
|
||||||
- name: Create GitHub Release
|
- name: Create GitHub Release
|
||||||
uses: emsesp/action-automatic-releases@v1.0.0
|
uses: emsesp/action-automatic-releases@v1.0.0
|
||||||
|
|||||||
6
.github/workflows/stale_issues.yml
vendored
6
.github/workflows/stale_issues.yml
vendored
@@ -1,4 +1,8 @@
|
|||||||
name: "Mark or close stale issues and PRs"
|
name: "Mark or close stale issues and PRs"
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
issues: write
|
||||||
|
pull-requests: write
|
||||||
|
|
||||||
on:
|
on:
|
||||||
schedule:
|
schedule:
|
||||||
@@ -8,7 +12,7 @@ jobs:
|
|||||||
stale:
|
stale:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/stale@v9
|
- uses: actions/stale@v10
|
||||||
with:
|
with:
|
||||||
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
days-before-stale: 30
|
days-before-stale: 30
|
||||||
|
|||||||
33
.github/workflows/test_release.yml
vendored
33
.github/workflows/test_release.yml
vendored
@@ -4,7 +4,7 @@ on:
|
|||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- 'dev2'
|
- 'test'
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
contents: read
|
contents: read
|
||||||
@@ -18,17 +18,17 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
|
|
||||||
- name: Install python 3.13
|
- name: Install python 3.13
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v6
|
||||||
with:
|
with:
|
||||||
python-version: '3.13'
|
python-version: '3.13'
|
||||||
|
|
||||||
- name: Install Node.js 22
|
- name: Install Node.js 24
|
||||||
uses: actions/setup-node@v4
|
uses: actions/setup-node@v6
|
||||||
with:
|
with:
|
||||||
node-version: 22
|
node-version: 24
|
||||||
|
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v5
|
||||||
|
|
||||||
- name: Enable Corepack
|
- name: Enable Corepack
|
||||||
run: corepack enable pnpm
|
run: corepack enable pnpm
|
||||||
@@ -45,20 +45,21 @@ jobs:
|
|||||||
pip install -U platformio
|
pip install -U platformio
|
||||||
python -m pip install intelhex
|
python -m pip install intelhex
|
||||||
|
|
||||||
- name: Build the WebUI
|
- name: Build webUI
|
||||||
run: |
|
run: |
|
||||||
cd interface
|
platformio run -e build_webUI
|
||||||
pnpm install
|
|
||||||
pnpm typesafe-i18n --no-watch
|
|
||||||
sed -i "s/= 'pl'/= 'en'/" ./src/i18n/i18n-util.ts
|
|
||||||
pnpm build
|
|
||||||
pnpm webUI
|
|
||||||
|
|
||||||
- name: Build all target environments
|
- name: Build modbus
|
||||||
|
run: |
|
||||||
|
platformio run -e build_modbus
|
||||||
|
|
||||||
|
- name: Build standalone
|
||||||
|
run: |
|
||||||
|
platformio run -e build_standalone
|
||||||
|
|
||||||
|
- name: Build all PIO target environments, from default_envs
|
||||||
run: |
|
run: |
|
||||||
platformio run
|
platformio run
|
||||||
env:
|
|
||||||
NO_BUILD_WEBUI: true
|
|
||||||
|
|
||||||
- name: Create GitHub Release
|
- name: Create GitHub Release
|
||||||
id: 'automatic_releases'
|
id: 'automatic_releases'
|
||||||
|
|||||||
5
.gitignore
vendored
5
.gitignore
vendored
@@ -2,7 +2,6 @@
|
|||||||
.vscode/c_cpp_properties.json
|
.vscode/c_cpp_properties.json
|
||||||
.vscode/extensions.json
|
.vscode/extensions.json
|
||||||
.vscode/launch.json
|
.vscode/launch.json
|
||||||
.vscode/settings.json
|
|
||||||
|
|
||||||
# c++ compiling
|
# c++ compiling
|
||||||
.clang_complete
|
.clang_complete
|
||||||
@@ -72,3 +71,7 @@ CMakeLists.txt
|
|||||||
logs/*
|
logs/*
|
||||||
sdkconfig.*
|
sdkconfig.*
|
||||||
sdkconfig_tasmota_esp32
|
sdkconfig_tasmota_esp32
|
||||||
|
pnpm-lock.yaml
|
||||||
|
.cache/
|
||||||
|
interface/.tsbuildinfo
|
||||||
|
test/test_api/package-lock.json
|
||||||
|
|||||||
116
CHANGELOG.md
116
CHANGELOG.md
@@ -5,6 +5,120 @@ All notable changes to this project will be documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## [3.8.1] 11 January 2026
|
||||||
|
|
||||||
|
## Added
|
||||||
|
|
||||||
|
- update time saved in nvs
|
||||||
|
- heatpump entities [#2883](https://github.com/emsesp/EMS-ESP32/issues/2883)
|
||||||
|
- HA input number format (mode) selectable box/slider (slider for max range 100) [#2900](https://github.com/emsesp/EMS-ESP32/discussions/2900)
|
||||||
|
|
||||||
|
## Fixed
|
||||||
|
|
||||||
|
- fix EMS bus disconnected errors on some systems [#2881](https://github.com/emsesp/EMS-ESP32/issues/2881)
|
||||||
|
- selflowtemp fix [#2876](https://github.com/emsesp/EMS-ESP32/issues/2876)
|
||||||
|
- updated valid GPIOs for ESP32S2, ESP32S3 and ESP32 that caused custom systems to block gpios [#2887](https://github.com/emsesp/EMS-ESP32/issues/2887)
|
||||||
|
- Junkers wwcharge offset [#2860](https://github.com/emsesp/EMS-ESP32/issues/2860)
|
||||||
|
- fixed minflowtemp [#2890](https://github.com/emsesp/EMS-ESP32/issues/2890)
|
||||||
|
- don't add HA uom/classes for bool values [#2885](https://github.com/emsesp/EMS-ESP32/issues/2885)
|
||||||
|
- fixed missing progress bar on web firmware uploads
|
||||||
|
|
||||||
|
## Changed
|
||||||
|
|
||||||
|
- snapshot gpios stored in temporary ram
|
||||||
|
- GPIOs stored along with the name and reported in log if conflicting
|
||||||
|
- free GPIOs depend on board profile [#2901](https://github.com/emsesp/EMS-ESP32/issues/2901)
|
||||||
|
- prefer PSram for mqtt queue [#2889](https://github.com/emsesp/EMS-ESP32/issues/2889)
|
||||||
|
- day schedule defult to all days, no day selected is not allowed
|
||||||
|
- board profile `CUSTOM` can only be selected in developer mode
|
||||||
|
- mqtt sends round values without decimals (`28` instead of `28.0`)
|
||||||
|
|
||||||
|
|
||||||
|
## [3.8.0] 31 December 2025
|
||||||
|
|
||||||
|
## Added
|
||||||
|
|
||||||
|
- analogsensor types: NTC and RGB-Led
|
||||||
|
- Flag for HMC310 [#2465](https://github.com/emsesp/EMS-ESP32/issues/2465)
|
||||||
|
- boiler auxheatersource [#2489](https://github.com/emsesp/EMS-ESP32/discussions/2489)
|
||||||
|
- thermostat last error for RC100/300 [#2501](https://github.com/emsesp/EMS-ESP32/issues/2501)
|
||||||
|
- boiler 0xC6 telegram [#1963](https://github.com/emsesp/EMS-ESP32/issues/1963)
|
||||||
|
- CS6800i changes [#2448](https://github.com/emsesp/EMS-ESP32/issues/2448), [#2449](https://github.com/emsesp/EMS-ESP32/issues/2449)
|
||||||
|
- charging pump [#2544](https://github.com/emsesp/EMS-ESP32/issues/2544)
|
||||||
|
- hybrid CSH5800iG [#2569](https://github.com/emsesp/EMS-ESP32/issues/2569)
|
||||||
|
- added EMS Device details to Home Assistant MQTT Discovery
|
||||||
|
- disinfection command [#2601](https://github.com/emsesp/EMS-ESP32/issues/2601)
|
||||||
|
- added new board profile for upcoming BBQKees E32V2.2
|
||||||
|
- set differential pressure entity in Mixer device
|
||||||
|
- set set climate action cooling/heating in HA [#2583](https://github.com/emsesp/EMS-ESP32/issues/2583)
|
||||||
|
- Internal sensors of E32V2_2
|
||||||
|
- FW200 display options [#2610](https://github.com/emsesp/EMS-ESP32/discussions/2610)
|
||||||
|
- CR11 mode settings OFF/MANUAL depends on selTemp [#2437](https://github.com/emsesp/EMS-ESP32/issues/2437)
|
||||||
|
- implemented eFuse settings for BBQKees boards to store model type and ESP chipset
|
||||||
|
- analogsensors for pulse output [#2624](https://github.com/emsesp/EMS-ESP32/discussions/2624)
|
||||||
|
- analogsensors frequency input [#2631](https://github.com/emsesp/EMS-ESP32/discussions/2631)
|
||||||
|
- SRC plus thermostats [#2636](https://github.com/emsesp/EMS-ESP32/issues/2636)
|
||||||
|
- Greenstar 2000 [#2645](https://github.com/emsesp/EMS-ESP32/issues/2645)
|
||||||
|
- RC3xx `dhw modetype` [#2659](https://github.com/emsesp/EMS-ESP32/discussions/2659)
|
||||||
|
- new boiler entities VR0,VR1, compressor speed [#2669](https://github.com/emsesp/EMS-ESP32/issues/2669)
|
||||||
|
- solar temperature TS16 [#2690](https://github.com/emsesp/EMS-ESP32/issues/2690)
|
||||||
|
- pumpmode enum for HT3 boilers, add commands for manual defrost, chimneysweeper [#2727](https://github.com/emsesp/EMS-ESP32/issues/2727)
|
||||||
|
- pid settings [#2735](https://github.com/emsesp/EMS-ESP32/issues/2735)
|
||||||
|
- refresh MQTT button added to MQTT Settings page
|
||||||
|
- heating assistance, rounding custum settings [#2763](https://github.com/emsesp/EMS-ESP32/discussions/2763)
|
||||||
|
- added counter 0..2 for short pulses, high frequency [#2758](https://github.com/emsesp/EMS-ESP32/issues/2758)
|
||||||
|
- added LWT (Last Will and Testament) to MQTT entities in Home Assistant
|
||||||
|
- added api/metrics endpoint for prometheus integration by @gr3enk [#2774](https://github.com/emsesp/EMS-ESP32/pull/2774)
|
||||||
|
- added RTL8201 to eth phy list [#2800](https://github.com/emsesp/EMS-ESP32/issues/2800)
|
||||||
|
- added partitions to Web UI Version page, so previous firmware versions can be installed [#2837](https://github.com/emsesp/EMS-ESP32/issues/2837)
|
||||||
|
- button pressures show LED. On a long press (10 seconds) the LED flashes for 5 seconds to indicate a factory reset is about to happen. [#2848](https://github.com/emsesp/EMS-ESP32/issues/2848)
|
||||||
|
- added `txpause` command to pause the TX, by setting Txmode to 0 (disabled) [#2850](https://github.com/emsesp/EMS-ESP32/issues/2850)
|
||||||
|
|
||||||
|
## Fixed
|
||||||
|
|
||||||
|
- dhw/switchtime [#2490](https://github.com/emsesp/EMS-ESP32/issues/2490)
|
||||||
|
- switch to secure mqtt [#2492](https://github.com/emsesp/EMS-ESP32/issues/2492)
|
||||||
|
- update link buttons [#2497](https://github.com/emsesp/EMS-ESP32/issues/2497)
|
||||||
|
- refresh scheduler states [#2502](https://github.com/emsesp/EMS-ESP32/discussions/2502)
|
||||||
|
- also rebuild HA config on mqtt connect for scheduler, custom and shower
|
||||||
|
- FB100 controls the hc, not the master [#2510](https://github.com/emsesp/EMS-ESP32/issues/2510)
|
||||||
|
- IPM DHW module, [#2524](https://github.com/emsesp/EMS-ESP32/issues/2524)
|
||||||
|
- charge optimization [#2543](https://github.com/emsesp/EMS-ESP32/issues/2543)
|
||||||
|
- shower active state retained, shows correctly in HA
|
||||||
|
- MQTT Command Topic with slashes [#2571](https://github.com/emsesp/EMS-ESP32/issues/2571)
|
||||||
|
- Add pulsed water meter input to V1.3 gateway with Lilygo S3 [#2550](https://github.com/emsesp/EMS-ESP32/issues/2550)
|
||||||
|
- fix missing long 10-second press of Button to perform a factory reset
|
||||||
|
- fix wwMaxPower on Junkers ZBS14 [#2609](https://github.com/emsesp/EMS-ESP32/issues/2609)
|
||||||
|
- ventilation bypass state from telegram 0x55C [#1197](https://github.com/emsesp/EMS-ESP32/issues/1197)
|
||||||
|
- set selflowtemp for ems+ boilers [#2641](https://github.com/emsesp/EMS-ESP32/discussions/2641)
|
||||||
|
- syslog timestamp [#2704](https://github.com/emsesp/EMS-ESP32/issues/2704)
|
||||||
|
- fixed FS format command [#2720](https://github.com/emsesp/EMS-ESP32/discussions/2720)
|
||||||
|
- dhw priority setting to boiler and mixer, telegrams 0x2CC, 0x2CD, etc.
|
||||||
|
- check for valid GPIOs when board profile is changed [#2841](https://github.com/emsesp/EMS-ESP32/issues/2841)
|
||||||
|
|
||||||
|
## Changed
|
||||||
|
|
||||||
|
- show console log with ISO date/time [#2533](https://github.com/emsesp/EMS-ESP32/discussions/2533)
|
||||||
|
- removed ESP32 CPU temperature
|
||||||
|
- updated core libraries like AsyncTCP, AsyncWebServer and Modbus
|
||||||
|
- remove command `scan deep`
|
||||||
|
- ignore repeated `forceheatingoff` commands [#2641](https://github.com/emsesp/EMS-ESP32/discussions/2641)
|
||||||
|
- optimized web for better performance by adding lazy loading and caching
|
||||||
|
- internal system analog sensors (core_voltage, supply_voltage and gateway_temperature) cannot be accidentally removed
|
||||||
|
- double click button reconnects EMS-ESP to AP
|
||||||
|
- place system message command in side scheduler loop to reduce stack memory usage by 2KB
|
||||||
|
- syslog mark interval set to 1 hour
|
||||||
|
- handle process_telegram in oneloop
|
||||||
|
- improved GPIO validation for Analog Sensors and System GPIOs
|
||||||
|
- entities with no values are greyed out in the Web UI in the Customization page
|
||||||
|
- added System Status to Web Status page
|
||||||
|
- show number on entities and supported languages in log on boot
|
||||||
|
- on tx read fail delay the 3rd. retry 2 sec
|
||||||
|
- move vectors and lists to PSRAM
|
||||||
|
- removed unused last topic/payload echo-check
|
||||||
|
- added Home Assistant device details to MQTT Discovery for all devices
|
||||||
|
- device_class and state_class changes for HA MQTT Discovery [#2825](https://github.com/emsesp/EMS-ESP32/issues/2825)
|
||||||
|
|
||||||
## [3.7.2] 22 March 2025
|
## [3.7.2] 22 March 2025
|
||||||
|
|
||||||
## Added
|
## Added
|
||||||
@@ -85,7 +199,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
- The automatically generated temperature sensor ID has replaced dashes (`-`) with underscores (`_`) to be compatible with Home Assistant.
|
- The automatically generated temperature sensor ID has replaced dashes (`-`) with underscores (`_`) to be compatible with Home Assistant.
|
||||||
- `api/system/info` has it's JSON key names changed to camelCase syntax.
|
- `api/system/info` has it's JSON key names changed to camelCase syntax.
|
||||||
|
|
||||||
For more details go to [docs.emsesp.org](https://docs.emsesp.org/).
|
For more details go to [emsesp.org](https://emsesp.org/).
|
||||||
|
|
||||||
## Added
|
## Added
|
||||||
|
|
||||||
|
|||||||
@@ -1,56 +1,33 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
For more details go to [docs.emsesp.org](https://docs.emsesp.org/).
|
For more details go to [emsesp.org](https://emsesp.org/).
|
||||||
|
|
||||||
## [3.7.3]
|
## [3.8.2]
|
||||||
|
|
||||||
## Added
|
## Added
|
||||||
|
|
||||||
- analogsensor types: NTC and RGB-Led
|
- comfortpoint for BC400 [#2935](https://github.com/emsesp/EMS-ESP32/issues/2935)
|
||||||
- Flag for HMC310 [#2465](https://github.com/emsesp/EMS-ESP32/issues/2465)
|
- customize device brand [#2784](https://github.com/emsesp/EMS-ESP32/issues/2784)
|
||||||
- boiler auxheatersource [#2489](https://github.com/emsesp/EMS-ESP32/discussions/2489)
|
- set model for ems-esp devices temperature, analog, etc. [#2958](https://github.com/emsesp/EMS-ESP32/discussions/2958)
|
||||||
- thermostat last error for RC100/300 [#2501](https://github.com/emsesp/EMS-ESP32/issues/2501)
|
- prometheus metrics for temperature/analog/scheduler/custom [#2962](https://github.com/emsesp/EMS-ESP32/issues/2962)
|
||||||
- boiler 0xC6 telegram [#1963](https://github.com/emsesp/EMS-ESP32/issues/1963)
|
- boiler pumpkick [#2965](https://github.com/emsesp/EMS-ESP32/discussions/2965)
|
||||||
- CS6800i changes [#2448](https://github.com/emsesp/EMS-ESP32/issues/2448), [#2449](https://github.com/emsesp/EMS-ESP32/issues/2449)
|
- heatpump reset [#2933](https://github.com/emsesp/EMS-ESP32/issues/2933)
|
||||||
- charging pump [#2544](https://github.com/emsesp/EMS-ESP32/issues/2544)
|
- e-mail notification using ReadyMail Client
|
||||||
- hybrid CSH5800iG [#2569](https://github.com/emsesp/EMS-ESP32/issues/2569)
|
- 2.nd freshwater module (dhw4) [#2991](https://github.com/emsesp/EMS-ESP32/issues/2991)
|
||||||
- add EMS Device details to Home Assistant MQTT Discovery
|
|
||||||
- disinfection command [#2601](https://github.com/emsesp/EMS-ESP32/issues/2601)
|
|
||||||
- added new board profile for upcoming BBQKees E32V2.2
|
|
||||||
- set differential pressure entity in Mixer device
|
|
||||||
- set set climate action cooling/heating in HA [#2583](https://github.com/emsesp/EMS-ESP32/issues/2583)
|
|
||||||
- Internal sensors of E32V2_2
|
|
||||||
- FW200 display options [#2610](https://github.com/emsesp/EMS-ESP32/discussions/2610)
|
|
||||||
- CR11 mode settings OFF/MANUAL depends on selTemp [#2437](https://github.com/emsesp/EMS-ESP32/issues/2437)
|
|
||||||
- Fuse settings for BBQKees boards
|
|
||||||
- Analogsensors for pulse output [#2624](https://github.com/emsesp/EMS-ESP32/discussions/2624)
|
|
||||||
- Analogsensors frequency input [#2631](https://github.com/emsesp/EMS-ESP32/discussions/2631)
|
|
||||||
- SRC plus thermostats [#2636](https://github.com/emsesp/EMS-ESP32/issues/2636)
|
|
||||||
- Greenstar 2000 [#2645](https://github.com/emsesp/EMS-ESP32/issues/2645)
|
|
||||||
- RC3xx `dhw modetype` [#2659](https://github.com/emsesp/EMS-ESP32/discussions/2659)
|
|
||||||
|
|
||||||
## Fixed
|
## Fixed
|
||||||
|
|
||||||
- dhw/switchtime [#2490](https://github.com/emsesp/EMS-ESP32/issues/2490)
|
- SRC climate creation [#2936](https://github.com/emsesp/EMS-ESP32/issues/2936) and [#2960](https://github.com/emsesp/EMS-ESP32/issues/2960)
|
||||||
- switch to secure mqtt [#2492](https://github.com/emsesp/EMS-ESP32/issues/2492)
|
|
||||||
- update link buttons [#2497](https://github.com/emsesp/EMS-ESP32/issues/2497)
|
|
||||||
- refresh scheduler states [#2502](https://github.com/emsesp/EMS-ESP32/discussions/2502)
|
|
||||||
- also rebuild HA config on mqtt connect for scheduler, custom and shower
|
|
||||||
- FB100 controls the hc, not the master [#2510](https://github.com/emsesp/EMS-ESP32/issues/2510)
|
|
||||||
- IPM DHW module, [#2524](https://github.com/emsesp/EMS-ESP32/issues/2524)
|
|
||||||
- charge optimization [#2543](https://github.com/emsesp/EMS-ESP32/issues/2543)
|
|
||||||
- shower active state retained, shows correctly in HA
|
|
||||||
- MQTT Command Topic with slashes [#2571](https://github.com/emsesp/EMS-ESP32/issues/2571)
|
|
||||||
- Add pulsed water meter input to V1.3 gateway with Lilygo S3 [#2550](https://github.com/emsesp/EMS-ESP32/issues/2550)
|
|
||||||
- fix missing long 10-second press of Button to perform a factory reset
|
|
||||||
- fix wwMaxPower on Junkers ZBS14 [#2609](https://github.com/emsesp/EMS-ESP32/issues/2609)
|
|
||||||
- ventilation bypass state from telegram 0x55C [#1197](https://github.com/emsesp/EMS-ESP32/issues/1197)
|
|
||||||
- set selflowtemp for ems+ boilers [#2641](https://github.com/emsesp/EMS-ESP32/discussions/2641)
|
|
||||||
|
|
||||||
## Changed
|
## Changed
|
||||||
|
|
||||||
- show console log with ISO date/time [#2533](https://github.com/emsesp/EMS-ESP32/discussions/2533)
|
- weblogbuffer up to 1000 messages with PSRAM, mentioned in [#2933](https://github.com/emsesp/EMS-ESP32/issues/2933)
|
||||||
- remove ESP32 CPU temperature
|
- validate custom entity writes, [#2931](https://github.com/emsesp/EMS-ESP32/issues/2931)
|
||||||
- updated core libraries like AsyncTCP, AsyncWebServer and Modbus
|
- remove wrong burnMinPower [#2918](https://github.com/emsesp/EMS-ESP32/issues/2918)
|
||||||
- remove command `scan deep`
|
- store scheduler active state to nvs [#2946](https://github.com/emsesp/EMS-ESP32/discussions/2946)
|
||||||
- ignore repeated `forceheatingoff` commands [#2641](https://github.com/emsesp/EMS-ESP32/discussions/2641)
|
- translated modes `heat` and `eco` for HA-climate mode-str-tpl
|
||||||
|
- support `minflowtemp` and `baseflowtemp` [#2969](https://github.com/emsesp/EMS-ESP32/discussions/2969)
|
||||||
|
- update version if it is 00.00 in first read [#2981](https://github.com/emsesp/EMS-ESP32/issues/2981)
|
||||||
|
- device class for % values [#2980](https://github.com/emsesp/EMS-ESP32/issues/2980)
|
||||||
|
- use tasmota core 2026.03.30
|
||||||
|
- secure mqtt uses ESP_SSLClient
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ Everybody is welcome and invited to contribute to the EMS-ESP Project by:
|
|||||||
|
|
||||||
- providing Pull Requests (Features, Fixes, suggestions)
|
- providing Pull Requests (Features, Fixes, suggestions)
|
||||||
- testing new released features and report issues on your EMS equipment
|
- testing new released features and report issues on your EMS equipment
|
||||||
- contributing to missing [documentation](https://docs.emsesp.org)
|
- contributing to missing [documentation](https://emsesp.org)
|
||||||
|
|
||||||
This document describes rules that are in effect for this repository, meant for handling issues by contributors in the issue tracker and PRs.
|
This document describes rules that are in effect for this repository, meant for handling issues by contributors in the issue tracker and PRs.
|
||||||
|
|
||||||
|
|||||||
61
Makefile
61
Makefile
@@ -19,17 +19,20 @@ C = $(words $N)$(eval N := x $N)
|
|||||||
ECHO = python3 $(I)/scripts/echo_progress.py --stepno=$C --nsteps=$T
|
ECHO = python3 $(I)/scripts/echo_progress.py --stepno=$C --nsteps=$T
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# determine number of parallel compiles based on OS
|
# Optimize parallel build configuration
|
||||||
UNAME_S := $(shell uname -s)
|
UNAME_S := $(shell uname -s)
|
||||||
|
JOBS ?= 1
|
||||||
ifeq ($(UNAME_S),Linux)
|
ifeq ($(UNAME_S),Linux)
|
||||||
EXTRA_CPPFLAGS = -D LINUX
|
EXTRA_CPPFLAGS = -D LINUX
|
||||||
JOBS ?= $(shell nproc)
|
JOBS := $(shell nproc)
|
||||||
endif
|
endif
|
||||||
ifeq ($(UNAME_S),Darwin)
|
ifeq ($(UNAME_S),Darwin)
|
||||||
EXTRA_CPPFLAGS = -D OSX -Wno-tautological-constant-out-of-range-compare
|
EXTRA_CPPFLAGS = -D OSX -Wno-tautological-constant-out-of-range-compare
|
||||||
JOBS ?= $(shell sysctl -n hw.ncpu)
|
JOBS := $(shell sysctl -n hw.ncpu)
|
||||||
endif
|
endif
|
||||||
MAKEFLAGS += -j $(JOBS) -l $(JOBS)
|
|
||||||
|
# Set optimal parallel build settings
|
||||||
|
MAKEFLAGS += -j$(JOBS) -l$(shell echo $$(($(JOBS) * 2)))
|
||||||
|
|
||||||
# $(info Number of jobs: $(JOBS))
|
# $(info Number of jobs: $(JOBS))
|
||||||
|
|
||||||
@@ -45,7 +48,7 @@ MAKEFLAGS += -j $(JOBS) -l $(JOBS)
|
|||||||
TARGET := emsesp
|
TARGET := emsesp
|
||||||
BUILD := build
|
BUILD := build
|
||||||
SOURCES := src/core src/devices src/web src/test lib_standalone lib/semver lib/espMqttClient/src lib/espMqttClient/src/* lib/ArduinoJson/src lib/uuid-common/src lib/uuid-console/src lib/uuid-log/src lib/PButton
|
SOURCES := src/core src/devices src/web src/test lib_standalone lib/semver lib/espMqttClient/src lib/espMqttClient/src/* lib/ArduinoJson/src lib/uuid-common/src lib/uuid-console/src lib/uuid-log/src lib/PButton
|
||||||
INCLUDES := src/core src/devices src/web src/test lib/* lib_standalone lib/semver lib/espMqttClient/src lib/espMqttClient/src/Transport lib/ArduinoJson/src lib/uuid-common/src lib/uuid-console/src lib/uuid-log/src lib/uuid-telnet/src lib/uuid-syslog/src
|
INCLUDES := src/core src/devices src/web src/test lib_standalone lib/* lib/semver lib/espMqttClient/src lib/espMqttClient/src/Transport lib/ArduinoJson/src lib/uuid-common/src lib/uuid-console/src lib/uuid-log/src lib/uuid-telnet/src lib/uuid-syslog/src
|
||||||
LIBRARIES :=
|
LIBRARIES :=
|
||||||
|
|
||||||
CPPCHECK = cppcheck
|
CPPCHECK = cppcheck
|
||||||
@@ -62,9 +65,10 @@ CXX_STANDARD := -std=gnu++17
|
|||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
DEFINES += -DARDUINOJSON_ENABLE -DARDUINOJSON_ENABLE_ARDUINO_STRING -DARDUINOJSON_USE_DOUBLE=0
|
DEFINES += -DARDUINOJSON_ENABLE -DARDUINOJSON_ENABLE_ARDUINO_STRING -DARDUINOJSON_USE_DOUBLE=0
|
||||||
DEFINES += -DEMSESP_STANDALONE -DEMSESP_TEST -DEMSESP_DEBUG -DEMC_RX_BUFFER_SIZE=1500
|
DEFINES += -DEMSESP_STANDALONE -DEMSESP_TEST -DEMSESP_DEBUG -DEMC_RX_BUFFER_SIZE=1500
|
||||||
|
DEFINES += -DNO_TLS_SUPPORT
|
||||||
DEFINES += $(ARGS)
|
DEFINES += $(ARGS)
|
||||||
|
|
||||||
DEFAULTS = -DEMSESP_DEFAULT_LOCALE=\"en\" -DEMSESP_DEFAULT_TX_MODE=8 -DEMSESP_DEFAULT_VERSION=\"3.7.3-dev\" -DEMSESP_DEFAULT_BOARD_PROFILE=\"S32S3\"
|
DEFAULTS = -DEMSESP_DEFAULT_LOCALE=\"en\" -DEMSESP_DEFAULT_BOARD_PROFILE=\"S32S3\"
|
||||||
|
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Sources & Files
|
# Sources & Files
|
||||||
@@ -72,16 +76,25 @@ DEFAULTS = -DEMSESP_DEFAULT_LOCALE=\"en\" -DEMSESP_DEFAULT_TX_MODE=8 -DEMSESP_DE
|
|||||||
OUTPUT := $(CURDIR)/$(TARGET)
|
OUTPUT := $(CURDIR)/$(TARGET)
|
||||||
SYMBOLS := $(CURDIR)/$(BUILD)/$(TARGET).out
|
SYMBOLS := $(CURDIR)/$(BUILD)/$(TARGET).out
|
||||||
|
|
||||||
CSOURCES := $(foreach dir,$(SOURCES),$(wildcard $(dir)/*.c))
|
# Optimize source discovery - use shell find for better performance
|
||||||
CXXSOURCES := $(foreach dir,$(SOURCES),$(wildcard $(dir)/*.cpp))
|
CSOURCES := $(shell find $(SOURCES) -name "*.c" 2>/dev/null)
|
||||||
|
CXXSOURCES := $(shell find $(SOURCES) -name "*.cpp" 2>/dev/null)
|
||||||
|
|
||||||
|
# Exclude files not needed for standalone build, if they exist
|
||||||
|
CSOURCES := $(filter-out src/core/ModuleLibrary.c,$(CSOURCES))
|
||||||
|
CXXSOURCES := $(filter-out src/core/ModuleLibrary.cpp,$(CXXSOURCES))
|
||||||
|
|
||||||
OBJS := $(patsubst %,$(BUILD)/%.o,$(basename $(CSOURCES)) $(basename $(CXXSOURCES)))
|
OBJS := $(patsubst %,$(BUILD)/%.o,$(basename $(CSOURCES)) $(basename $(CXXSOURCES)))
|
||||||
DEPS := $(patsubst %,$(BUILD)/%.d,$(basename $(CSOURCES)) $(basename $(CXXSOURCES)))
|
DEPS := $(patsubst %,$(BUILD)/%.d,$(basename $(CSOURCES)) $(basename $(CXXSOURCES)))
|
||||||
|
|
||||||
INCLUDE += $(addprefix -I,$(foreach dir,$(INCLUDES), $(wildcard $(dir))))
|
# Optimize include path discovery
|
||||||
INCLUDE += $(addprefix -I,$(foreach dir,$(LIBRARIES),$(wildcard $(dir)/include)))
|
INCLUDE_DIRS := $(shell find $(INCLUDES) -type d 2>/dev/null)
|
||||||
|
LIBRARY_INCLUDES := $(shell find $(LIBRARIES) -name "include" -type d 2>/dev/null)
|
||||||
|
INCLUDE += $(addprefix -I,$(INCLUDE_DIRS) $(LIBRARY_INCLUDES))
|
||||||
|
|
||||||
LDLIBS += $(addprefix -L,$(foreach dir,$(LIBRARIES),$(wildcard $(dir)/lib)))
|
# Optimize library path discovery
|
||||||
|
LIBRARY_DIRS := $(shell find $(LIBRARIES) -name "lib" -type d 2>/dev/null)
|
||||||
|
LDLIBS += $(addprefix -L,$(LIBRARY_DIRS))
|
||||||
|
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Compiler & Linker
|
# Compiler & Linker
|
||||||
@@ -99,9 +112,11 @@ CXX := /usr/bin/g++
|
|||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
CPPFLAGS += $(DEFINES) $(DEFAULTS) $(INCLUDE)
|
CPPFLAGS += $(DEFINES) $(DEFAULTS) $(INCLUDE)
|
||||||
CPPFLAGS += -ggdb -g3 -MMD
|
CPPFLAGS += -ggdb -g3 -MMD
|
||||||
CPPFLAGS += -flto=auto -fno-lto
|
CPPFLAGS += -flto=auto
|
||||||
CPPFLAGS += -Wall -Wextra -Werror -Wswitch-enum
|
CPPFLAGS += -Wall -Wextra -Werror -Wswitch-enum
|
||||||
CPPFLAGS += -Wno-unused-parameter -Wno-missing-braces -Wno-vla-cxx-extension
|
CPPFLAGS += -Wno-unused-parameter -Wno-missing-braces -Wno-vla-cxx-extension
|
||||||
|
CPPFLAGS += -ffunction-sections -fdata-sections -fno-exceptions -fno-rtti -fno-threadsafe-statics
|
||||||
|
CPPFLAGS += -Os -DNDEBUG
|
||||||
|
|
||||||
CPPFLAGS += $(EXTRA_CPPFLAGS)
|
CPPFLAGS += $(EXTRA_CPPFLAGS)
|
||||||
|
|
||||||
@@ -122,11 +137,13 @@ else
|
|||||||
LD := $(CXX)
|
LD := $(CXX)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
#DEPFLAGS += -MF $(BUILD)/$*.d
|
# Dependency file generation
|
||||||
|
DEPFLAGS += -MF $(BUILD)/$*.d -MT $@
|
||||||
|
|
||||||
LINK.o = $(LD) $(LDFLAGS) $(LDLIBS) $^ -o $@
|
LINK.o = $(LD) $(LDFLAGS) $(LDLIBS) $^ -o $@
|
||||||
COMPILE.c = $(CC) $(C_STANDARD) $(CFLAGS) $(DEPFLAGS) -c $< -o $@
|
COMPILE.c = $(CC) $(C_STANDARD) $(CFLAGS) $(DEPFLAGS) -c $< -o $@
|
||||||
COMPILE.cpp = $(CXX) $(CXX_STANDARD) $(CXXFLAGS) $(DEPFLAGS) -c $< -o $@
|
COMPILE.cpp = $(CXX) $(CXX_STANDARD) $(CXXFLAGS) $(DEPFLAGS) -c $< -o $@
|
||||||
|
COMPILE.s = $(CC) $(CFLAGS) $(DEPFLAGS) -c $< -o $@
|
||||||
|
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Special Built-in Target
|
# Special Built-in Target
|
||||||
@@ -139,7 +156,10 @@ COMPILE.cpp = $(CXX) $(CXX_STANDARD) $(CXXFLAGS) $(DEPFLAGS) -c $< -o $@
|
|||||||
.SUFFIXES:
|
.SUFFIXES:
|
||||||
.INTERMEDIATE:
|
.INTERMEDIATE:
|
||||||
.PRECIOUS: $(OBJS) $(DEPS)
|
.PRECIOUS: $(OBJS) $(DEPS)
|
||||||
.PHONY: all clean help
|
.PHONY: all clean help cppcheck run
|
||||||
|
|
||||||
|
# Enable second expansion for more flexible rules
|
||||||
|
.SECONDEXPANSION:
|
||||||
|
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Targets
|
# Targets
|
||||||
@@ -154,7 +174,6 @@ $(OUTPUT): $(OBJS)
|
|||||||
@mkdir -p $(@D)
|
@mkdir -p $(@D)
|
||||||
@$(ECHO) Linking $@
|
@$(ECHO) Linking $@
|
||||||
$(LINK.o)
|
$(LINK.o)
|
||||||
$(SYMBOLS.out)
|
|
||||||
|
|
||||||
$(BUILD)/%.o: %.c
|
$(BUILD)/%.o: %.c
|
||||||
@mkdir -p $(@D)
|
@mkdir -p $(@D)
|
||||||
@@ -168,6 +187,7 @@ $(BUILD)/%.o: %.cpp
|
|||||||
|
|
||||||
$(BUILD)/%.o: %.s
|
$(BUILD)/%.o: %.s
|
||||||
@mkdir -p $(@D)
|
@mkdir -p $(@D)
|
||||||
|
@$(ECHO) Compiling $@
|
||||||
@$(COMPILE.s)
|
@$(COMPILE.s)
|
||||||
|
|
||||||
cppcheck: $(SOURCES)
|
cppcheck: $(SOURCES)
|
||||||
@@ -182,8 +202,15 @@ clean:
|
|||||||
@$(RM) -rf $(BUILD) $(OUTPUT)
|
@$(RM) -rf $(BUILD) $(OUTPUT)
|
||||||
|
|
||||||
help:
|
help:
|
||||||
@echo available targets: all run clean
|
@echo "Available targets:"
|
||||||
@echo $(OUTPUT)
|
@echo " all - Build the project (default)"
|
||||||
|
@echo " run - Build and run the executable"
|
||||||
|
@echo " clean - Remove build artifacts"
|
||||||
|
@echo " cppcheck - Run static analysis"
|
||||||
|
@echo " help - Show this help message"
|
||||||
|
@echo ""
|
||||||
|
@echo "Output: $(OUTPUT)"
|
||||||
|
@echo "Jobs: $(JOBS)"
|
||||||
|
|
||||||
-include $(DEPS)
|
-include $(DEPS)
|
||||||
|
|
||||||
|
|||||||
30
README.md
30
README.md
@@ -15,10 +15,10 @@
|
|||||||
<a href="https://github.com/emsesp/EMS-ESP32/blob/dev/CONTRIBUTING.md">
|
<a href="https://github.com/emsesp/EMS-ESP32/blob/dev/CONTRIBUTING.md">
|
||||||
<img src="https://img.shields.io/badge/Contribute-ff4785?style=for-the-badge&logo=git&logoColor=white" alt="Contribute" />
|
<img src="https://img.shields.io/badge/Contribute-ff4785?style=for-the-badge&logo=git&logoColor=white" alt="Contribute" />
|
||||||
</a>
|
</a>
|
||||||
<a href="https://docs.emsesp.org">
|
<a href="https://emsesp.org">
|
||||||
<img src="https://img.shields.io/badge/Documentation-0077b5?style=for-the-badge&logo=googledocs&logoColor=white" alt="Guides" />
|
<img src="https://img.shields.io/badge/Documentation-0077b5?style=for-the-badge&logo=googledocs&logoColor=white" alt="Guides" />
|
||||||
</a>
|
</a>
|
||||||
<a href="https://discord.gg/3J3GgnzpyT">
|
<a href="https://discord.gg/GP9DPSgeJq">
|
||||||
<img src="https://img.shields.io/badge/Discord-7289da?style=for-the-badge&logo=discord&logoColor=white" alt="Discord" />
|
<img src="https://img.shields.io/badge/Discord-7289da?style=for-the-badge&logo=discord&logoColor=white" alt="Discord" />
|
||||||
</a>
|
</a>
|
||||||
<a href="https://github.com/emsesp/EMS-ESP32/blob/main/CHANGELOG.md">
|
<a href="https://github.com/emsesp/EMS-ESP32/blob/main/CHANGELOG.md">
|
||||||
@@ -32,7 +32,8 @@
|
|||||||
[](https://sonarcloud.io/summary/new_code?id=emsesp_EMS-ESP32)
|
[](https://sonarcloud.io/summary/new_code?id=emsesp_EMS-ESP32)
|
||||||
[](https://app.codacy.com/gh/emsesp/EMS-ESP32/dashboard?utm_source=gh&utm_medium=referral&utm_content=&utm_campaign=Badge_grade)
|
[](https://app.codacy.com/gh/emsesp/EMS-ESP32/dashboard?utm_source=gh&utm_medium=referral&utm_content=&utm_campaign=Badge_grade)
|
||||||
[](https://github.com/emsesp/EMS-ESP32/releases)
|
[](https://github.com/emsesp/EMS-ESP32/releases)
|
||||||
[](https://discord.gg/3J3GgnzpyT)
|
[](https://discord.gg/GP9DPSgeJq)
|
||||||
|
[](https://deepwiki.com/emsesp/EMS-ESP32)
|
||||||
|
|
||||||
[](https://github.com/emsesp/EMS-ESP32/stargazers)
|
[](https://github.com/emsesp/EMS-ESP32/stargazers)
|
||||||
[](https://github.com/emsesp/EMS-ESP32/network)
|
[](https://github.com/emsesp/EMS-ESP32/network)
|
||||||
@@ -40,7 +41,8 @@
|
|||||||
|
|
||||||
**EMS-ESP** is an open-source firmware for the Espressif ESP32 microcontroller to communicate with **EMS** (Energy Management System) compatible equipment from manufacturers such as Bosch, Buderus, Nefit, Junkers, Worcester, Sieger, elm.leblanc and iVT.
|
**EMS-ESP** is an open-source firmware for the Espressif ESP32 microcontroller to communicate with **EMS** (Energy Management System) compatible equipment from manufacturers such as Bosch, Buderus, Nefit, Junkers, Worcester, Sieger, elm.leblanc and iVT.
|
||||||
|
|
||||||
It requires a small circuit to interface with the EMS bus which can be purchased from <https://bbqkees-electronics.nl> or custom built.
|
It requires a small circuit to interface with the EMS bus which can be purchased from <https://bbqkees-electronics.nl>. These gateways are tested thoroughly and certified to work with EMS-ESP.
|
||||||
|
|
||||||
|
|
||||||
## 📦 **Key Features**
|
## 📦 **Key Features**
|
||||||
|
|
||||||
@@ -60,35 +62,39 @@ It requires a small circuit to interface with the EMS bus which can be purchased
|
|||||||
|
|
||||||
## 🚀 **Installing**
|
## 🚀 **Installing**
|
||||||
|
|
||||||
Head over to [download.emsesp.org](https://download.emsesp.org) for instructions on how to install EMS-ESP. There is also further details on which boards are supported in [this section](https://docs.emsesp.org/Installing/) of the documentation.
|
Head over to the [Installation Guide](https://emsesp.org/Installing) section of the documentation for instructions on how to install EMS-ESP.
|
||||||
|
|
||||||
## 📋 **Documentation**
|
## 📋 **Documentation**
|
||||||
|
|
||||||
Visit [emsesp.org](https://docs.emsesp.org) for more details on how to install and configure EMS-ESP. There is also a collection of Frequently Asked Questions and Troubleshooting tips with example customizations from the community.
|
Visit [emsesp.org](https://emsesp.org) for more details on how to setup and configure EMS-ESP. You'll also find more a collection of example configuarations, Frequently Asked Questions and Troubleshooting tips.
|
||||||
|
|
||||||
## 💬 **Getting Support**
|
## 💬 **Getting Support**
|
||||||
|
|
||||||
To chat with the community reach out on our [Discord Server](https://discord.gg/3J3GgnzpyT).
|
To chat with the community reach out on our [Discord Server](https://discord.gg/GP9DPSgeJq).
|
||||||
|
|
||||||
If you find an issue or have a request, see [here](https://docs.emsesp.org/Support/) on how to submit a bug report or feature request.
|
If you find an issue or have a request, see the [Getting Support](https://emsesp.org/Support/) section of the documentation. Note if you are using a non-BBQKees EMS gateway, you may need to contact the manufacturer for support.
|
||||||
|
|
||||||
## 🎥 **Live Demo**
|
## 🎥 **Live Demo**
|
||||||
|
|
||||||
For a live demo go to [demo.emsesp.org](https://demo.emsesp.org). Pick a language from the sign on page and log in with any username or password. Note not all features are operational as it's based on static data.
|
To see a live demo go to [demo.emsesp.org](https://demo.emsesp.org). Pick a language and use any username and password to log in. Note whast you're seeing is static example data so not all features are operational.
|
||||||
|
|
||||||
## 💖 **Contributors**
|
## 💖 **Contributors**
|
||||||
|
|
||||||
EMS-ESP is a project created by [proddy](https://github.com/proddy) and owned and maintained by both [proddy](https://github.com/proddy) and [MichaelDvP](https://github.com/MichaelDvP) with support from [BBQKees Electronics](https://bbqkees-electronics.nl).
|
EMS-ESP is a project originally created by [proddy](https://github.com/proddy) and maintained by the ems-esp community.
|
||||||
|
|
||||||
If you like **EMS-ESP**, please give it a ✨ on GitHub, or even better fork it and contribute. You can also offer a small donation. This is an open-source project maintained by volunteers, and your support is greatly appreciated.
|
If you like **EMS-ESP**, please give it a ✨ on GitHub, or even better fork it and contribute. You can also offer a small donation. This is an open-source project maintained by volunteers, and your support is greatly appreciated.
|
||||||
|
|
||||||
|
## 📦 **Building**
|
||||||
|
|
||||||
|
See the [Building the firmware](https://emsesp.org/Building) guide in the documentation for instructions on how to build EMS-ESP from this source code.
|
||||||
|
|
||||||
## 📢 **Libraries used**
|
## 📢 **Libraries used**
|
||||||
|
|
||||||
- [esp8266-react](https://github.com/rjwats/esp8266-react) by @rjwats for the core framework that provides the Web UI, which has been heavily modified
|
- [esp8266-react](https://github.com/rjwats/esp8266-react) originally by @rjwats for the core framework that provides the Web UI, which has been heavily modified
|
||||||
- [uuid-\*](https://github.com/nomis/mcu-uuid-console) from @nomis. The console, syslog, telnet and logging are based off these awesome open source libraries
|
- [uuid-\*](https://github.com/nomis/mcu-uuid-console) from @nomis. The console, syslog, telnet and logging are based off these awesome open source libraries
|
||||||
- [ArduinoJson](https://github.com/bblanchon/ArduinoJson) for all the JSON processing
|
- [ArduinoJson](https://github.com/bblanchon/ArduinoJson) for all the JSON processing
|
||||||
- [espMqttClient](https://github.com/bertmelis/espMqttClient) for the MQTT client
|
- [espMqttClient](https://github.com/bertmelis/espMqttClient) for the MQTT client
|
||||||
- [ESPAsyncWebServer](https://github.com/ESP32Async/ESPAsyncWebServer) and [AsyncTCP](https://github.com/ESP32Async/AsyncTCP) for the Web server and TCP backends
|
- [ESPAsyncWebServer](https://github.com/ESP32Async/ESPAsyncWebServer) and [AsyncTCP](https://github.com/ESP32Async/AsyncTCP) for the Web server
|
||||||
|
|
||||||
## 📜 **License**
|
## 📜 **License**
|
||||||
|
|
||||||
|
|||||||
5
SECURITY.md
Normal file
5
SECURITY.md
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
# Security Policy
|
||||||
|
|
||||||
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
|
Please report any security vulnerabilities using the [Contact Form](https://emsesp.org/About/#-contact).
|
||||||
@@ -5,7 +5,7 @@
|
|||||||
},
|
},
|
||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": [
|
"extra_flags": [
|
||||||
"-DTASMOTA_SDK",
|
"-DNO_TLS_SUPPORT",
|
||||||
"-DARDUINO_LOLIN_C3_MINI",
|
"-DARDUINO_LOLIN_C3_MINI",
|
||||||
"-DARDUINO_USB_MODE=1",
|
"-DARDUINO_USB_MODE=1",
|
||||||
"-DARDUINO_USB_CDC_ON_BOOT=1"
|
"-DARDUINO_USB_CDC_ON_BOOT=1"
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": [
|
"extra_flags": [
|
||||||
"-DBOARD_HAS_PSRAM",
|
"-DBOARD_HAS_PSRAM",
|
||||||
"-DTASMOTA_SDK",
|
"-DNO_TLS_SUPPORT",
|
||||||
"-DARDUINO_USB_CDC_ON_BOOT=1",
|
"-DARDUINO_USB_CDC_ON_BOOT=1",
|
||||||
"-DARDUINO_USB_MODE=0"
|
"-DARDUINO_USB_MODE=0"
|
||||||
],
|
],
|
||||||
@@ -37,8 +37,8 @@
|
|||||||
"flash_size": "4MB",
|
"flash_size": "4MB",
|
||||||
"maximum_ram_size": 327680,
|
"maximum_ram_size": 327680,
|
||||||
"maximum_size": 4194304,
|
"maximum_size": 4194304,
|
||||||
"use_1200bps_touch": true,
|
"use_1200bps_touch": false,
|
||||||
"wait_for_upload_port": true,
|
"wait_for_upload_port": false,
|
||||||
"require_upload_port": true,
|
"require_upload_port": true,
|
||||||
"speed": 921600
|
"speed": 921600
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -21,11 +21,11 @@
|
|||||||
"arduino",
|
"arduino",
|
||||||
"espidf"
|
"espidf"
|
||||||
],
|
],
|
||||||
"name": "Espressif ESP32-S3 32M Flash OPI PSRAM, 4608KB Code/OTA, 2MB FS",
|
"name": "Tasmota ESP32-S3 32M Flash OPI PSRAM, 4608KB Code/OTA, 2MB FS",
|
||||||
"upload": {
|
"upload": {
|
||||||
"flash_size": "32MB",
|
"flash_size": "32MB",
|
||||||
"maximum_ram_size": 327680,
|
"maximum_ram_size": 327680,
|
||||||
"maximum_size": 16777216,
|
"maximum_size": 33554432,
|
||||||
"require_upload_port": true,
|
"require_upload_port": true,
|
||||||
"speed": 460800
|
"speed": 460800
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"build": {
|
"build": {
|
||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": "-DTASMOTA_SDK",
|
"extra_flags": "-DNO_TLS_SUPPORT",
|
||||||
"f_cpu": "240000000L",
|
"f_cpu": "240000000L",
|
||||||
"f_flash": "40000000L",
|
"f_flash": "40000000L",
|
||||||
"flash_mode": "dio",
|
"flash_mode": "dio",
|
||||||
@@ -19,7 +19,7 @@
|
|||||||
"arduino",
|
"arduino",
|
||||||
"espidf"
|
"espidf"
|
||||||
],
|
],
|
||||||
"name": "Espressif ESP32 16M Flash, 4608KB Code/OTA, 2MB FS",
|
"name": "Tasmota ESP32 16M Flash, 4608KB Code/OTA, 2MB FS",
|
||||||
"upload": {
|
"upload": {
|
||||||
"flash_size": "16MB",
|
"flash_size": "16MB",
|
||||||
"maximum_ram_size": 327680,
|
"maximum_ram_size": 327680,
|
||||||
|
|||||||
@@ -19,7 +19,7 @@
|
|||||||
"arduino",
|
"arduino",
|
||||||
"espidf"
|
"espidf"
|
||||||
],
|
],
|
||||||
"name": "Espressif ESP32 16M Flash DIO PSRAM, 4608KB Code/OTA, 2MB FS",
|
"name": "Tasmota ESP32 16M Flash DIO PSRAM, 4608KB Code/OTA, 2MB FS",
|
||||||
"upload": {
|
"upload": {
|
||||||
"flash_size": "16MB",
|
"flash_size": "16MB",
|
||||||
"maximum_ram_size": 327680,
|
"maximum_ram_size": 327680,
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"build": {
|
"build": {
|
||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": "-DTASMOTA_SDK",
|
"extra_flags": "-DNO_TLS_SUPPORT",
|
||||||
"f_cpu": "240000000L",
|
"f_cpu": "240000000L",
|
||||||
"f_flash": "40000000L",
|
"f_flash": "40000000L",
|
||||||
"flash_mode": "dio",
|
"flash_mode": "dio",
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
"build": {
|
"build": {
|
||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": [
|
"extra_flags": [
|
||||||
|
"-DNO_TLS_SUPPORT",
|
||||||
"-DARDUINO_XIAO_ESP32C6",
|
"-DARDUINO_XIAO_ESP32C6",
|
||||||
"-DARDUINO_USB_MODE=1",
|
"-DARDUINO_USB_MODE=1",
|
||||||
"-DARDUINO_USB_CDC_ON_BOOT=1"
|
"-DARDUINO_USB_CDC_ON_BOOT=1"
|
||||||
|
|||||||
@@ -9,6 +9,7 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"dictionaries": ["project-words"],
|
"dictionaries": ["project-words"],
|
||||||
|
"caseSensitive": false,
|
||||||
"ignorePaths": [
|
"ignorePaths": [
|
||||||
"node_modules",
|
"node_modules",
|
||||||
"compile_commands.json",
|
"compile_commands.json",
|
||||||
@@ -34,6 +35,10 @@
|
|||||||
"sdkconfig.*",
|
"sdkconfig.*",
|
||||||
"managed_components/**",
|
"managed_components/**",
|
||||||
"pnpm-*.yaml",
|
"pnpm-*.yaml",
|
||||||
"vite.config.ts"
|
"vite.config.ts",
|
||||||
|
"lib/esp32-psram/**",
|
||||||
|
"test/test_api/test_api.h",
|
||||||
|
"lib_standalone/**",
|
||||||
|
"lib/mbedtls_ssl/**"
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
11797
docs/dump_entities.csv
11797
docs/dump_entities.csv
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,233 @@
|
|||||||
|
telegram_type_id,name,is_fetched
|
||||||
|
0x04,UBAFactory,fetched
|
||||||
|
0x06,RCTime,
|
||||||
|
0x0A,EasyMonitor,fetched
|
||||||
|
0x10,UBAErrorMessage1,
|
||||||
|
0x11,UBAErrorMessage2,
|
||||||
|
0x12,RCErrorMessage,
|
||||||
|
0x13,RCErrorMessage2,
|
||||||
|
0x14,UBATotalUptime,fetched
|
||||||
|
0x15,UBAMaintenanceData,
|
||||||
|
0x16,UBAParameters,fetched
|
||||||
|
0x18,UBAMonitorFast,
|
||||||
|
0x19,UBAMonitorSlow,
|
||||||
|
0x1A,UBASetPoints,
|
||||||
|
0x1C,UBAMaintenanceStatus,
|
||||||
|
0x1E,HydrTemp,
|
||||||
|
0x23,JunkersSetMixer,fetched
|
||||||
|
0x27,UBASettingsWW,fetched
|
||||||
|
0x28,WeatherComp,fetched
|
||||||
|
0x2A,MC110Status,
|
||||||
|
0x2E,Meters,
|
||||||
|
0x33,UBAParameterWW,fetched
|
||||||
|
0x34,UBAMonitorWW,
|
||||||
|
0x35,UBAFlags,
|
||||||
|
0x37,WWSettings,fetched
|
||||||
|
0x38,WWTimer,fetched
|
||||||
|
0x39,WWCircTimer,fetched
|
||||||
|
0x3A,RC30WWSettings,fetched
|
||||||
|
0x3B,Energy,
|
||||||
|
0x3D,RC35Set,
|
||||||
|
0x3E,RC35Monitor,
|
||||||
|
0x3F,RC35Timer,
|
||||||
|
0x40,RC30Temp,
|
||||||
|
0x41,RC30Monitor,
|
||||||
|
0x42,RC35Timer2,
|
||||||
|
0x47,RC35Set,
|
||||||
|
0x48,RC35Monitor,
|
||||||
|
0x49,RC35Timer,
|
||||||
|
0x4C,RC35Timer2,
|
||||||
|
0x51,RC35Set,
|
||||||
|
0x52,RC35Monitor,
|
||||||
|
0x53,RC35Timer,
|
||||||
|
0x56,RC35Timer2,
|
||||||
|
0x5B,RC35Set,
|
||||||
|
0x5C,RC35Monitor,
|
||||||
|
0x5D,RC35Timer,
|
||||||
|
0x60,RC35Timer2,
|
||||||
|
0x96,SM10Config,fetched
|
||||||
|
0x97,SM10Monitor,
|
||||||
|
0x9C,WM10MonitorMessage,
|
||||||
|
0x9D,WM10SetMessage,
|
||||||
|
0xA2,RCError,
|
||||||
|
0xA3,RCOutdoorTemp,
|
||||||
|
0xA5,IBASettings,fetched
|
||||||
|
0xA7,RC30Set,
|
||||||
|
0xA9,RC30Vacation,fetched
|
||||||
|
0xAA,MMConfigMessage,fetched
|
||||||
|
0xAB,MMStatusMessage,
|
||||||
|
0xAC,MMSetMessage,
|
||||||
|
0xAF,RC20Remote,
|
||||||
|
0xB0,RC10Set,
|
||||||
|
0xB1,RC10Monitor,
|
||||||
|
0xBB,HybridSettings,fetched
|
||||||
|
0xBF,ErrorMessage,
|
||||||
|
0xC0,RCErrorMessage,
|
||||||
|
0xC2,UBAErrorMessage3,
|
||||||
|
0xC6,UBAErrorMessage3,
|
||||||
|
0xD1,UBAOutdoorTemp,
|
||||||
|
0xE3,UBAMonitorSlowPlus2,
|
||||||
|
0xE4,UBAMonitorFastPlus,
|
||||||
|
0xE5,UBAMonitorSlowPlus,
|
||||||
|
0xE6,UBAParametersPlus,fetched
|
||||||
|
0xE9,UBAMonitorWWPlus,
|
||||||
|
0xEA,UBAParameterWWPlus,fetched
|
||||||
|
0xEB,PumpKick,fetched
|
||||||
|
0x0101,ISM1Set,fetched
|
||||||
|
0x0103,ISM1StatusMessage,fetched
|
||||||
|
0x0104,ISM2StatusMessage,
|
||||||
|
0x010C,IPMStatusMessage,
|
||||||
|
0x011E,IPMTempMessage,
|
||||||
|
0x012E,HPEnergy1,
|
||||||
|
0x013B,HPEnergy2,
|
||||||
|
0x0165,JunkersSet,
|
||||||
|
0x0166,JunkersSet,
|
||||||
|
0x0167,JunkersSet,
|
||||||
|
0x0168,JunkersSet,
|
||||||
|
0x016E,Absent,fetched
|
||||||
|
0x016F,JunkersMonitor,
|
||||||
|
0x0170,JunkersMonitor,
|
||||||
|
0x0171,JunkersMonitor,
|
||||||
|
0x0172,JunkersMonitor,
|
||||||
|
0x0179,JunkersSet,
|
||||||
|
0x017A,JunkersSet,
|
||||||
|
0x017B,JunkersSet,
|
||||||
|
0x017C,JunkersSet,
|
||||||
|
0x01D3,JunkersDhw,fetched
|
||||||
|
0x023A,RC300OutdoorTemp,fetched
|
||||||
|
0x023E,PVSettings,fetched
|
||||||
|
0x0240,RC300Settings,fetched
|
||||||
|
0x0241,RC300Settings,fetched
|
||||||
|
0x0267,RC300Floordry,
|
||||||
|
0x0269,RC300Holiday,fetched
|
||||||
|
0x0291,HPMode,fetched
|
||||||
|
0x0292,HPMode,fetched
|
||||||
|
0x0293,HPMode,fetched
|
||||||
|
0x0294,HPMode,fetched
|
||||||
|
0x029B,RC300Curves,
|
||||||
|
0x029C,RC300Curves,
|
||||||
|
0x029D,RC300Curves,
|
||||||
|
0x029E,RC300Curves,
|
||||||
|
0x029F,RC300Curves,
|
||||||
|
0x02A0,RC300Curves,
|
||||||
|
0x02A1,RC300Curves,
|
||||||
|
0x02A2,RC300Curves,
|
||||||
|
0x02A5,RC300Monitor,fetched
|
||||||
|
0x02A6,CRFMonitor,
|
||||||
|
0x02A7,RC300Monitor,
|
||||||
|
0x02A8,CRFMonitor,
|
||||||
|
0x02A9,RC300Monitor,
|
||||||
|
0x02AA,RC300Monitor,
|
||||||
|
0x02AB,RC300Monitor,
|
||||||
|
0x02AC,RC300Monitor,
|
||||||
|
0x02AF,RC300Summer,
|
||||||
|
0x02B0,RC300Summer,
|
||||||
|
0x02B1,RC300Summer,
|
||||||
|
0x02B2,RC300Summer,
|
||||||
|
0x02B3,RC300Summer,
|
||||||
|
0x02B4,RC300Summer,
|
||||||
|
0x02B5,RC300Summer,
|
||||||
|
0x02B6,RC300Summer,
|
||||||
|
0x02B9,RC300Set,
|
||||||
|
0x02BA,RC300Set,
|
||||||
|
0x02BB,RC300Set,
|
||||||
|
0x02BC,RC300Set,
|
||||||
|
0x02BD,RC300Set,
|
||||||
|
0x02BE,RC300Set,
|
||||||
|
0x02BF,RC300Set,
|
||||||
|
0x02C0,RC300Set,
|
||||||
|
0x02CC,HPPressure,fetched
|
||||||
|
0x02CD,MMPLUSConfigMessage,
|
||||||
|
0x02D6,HPPump2,fetched
|
||||||
|
0x02D7,MMPLUSStatusMessage,
|
||||||
|
0x02E0,UBASetPoints,
|
||||||
|
0x02F5,RC300WWmode,fetched
|
||||||
|
0x02F6,RC300WW2mode,fetched
|
||||||
|
0x0313,MMPLUSConfigMessage_WWC,fetched
|
||||||
|
0x031B,RC300WWtemp,fetched
|
||||||
|
0x031D,RC300WWmode2,
|
||||||
|
0x031E,RC300WWmode2,
|
||||||
|
0x0331,MMPLUSStatusMessage_WWC,
|
||||||
|
0x0358,SM100SystemConfig,fetched
|
||||||
|
0x035A,SM100CircuitConfig,fetched
|
||||||
|
0x035C,SM100HeatAssist,fetched
|
||||||
|
0x035D,SM100Circuit2Config,fetched
|
||||||
|
0x035F,SM100Config1,fetched
|
||||||
|
0x0361,SM100Differential,fetched
|
||||||
|
0x0362,SM100Monitor,
|
||||||
|
0x0363,SM100Monitor2,
|
||||||
|
0x0364,SM100Status,
|
||||||
|
0x0366,SM100Config,
|
||||||
|
0x036A,SM100Status2,
|
||||||
|
0x0380,SM100CollectorConfig,fetched
|
||||||
|
0x038E,SM100Energy,fetched
|
||||||
|
0x0391,SM100Time,fetched
|
||||||
|
0x0421,RC300Set2,
|
||||||
|
0x0422,RC300Set2,
|
||||||
|
0x0423,RC300Set2,
|
||||||
|
0x0424,RC300Set2,
|
||||||
|
0x043F,CRHolidays,fetched
|
||||||
|
0x0467,HPSet,
|
||||||
|
0x0468,HPSet,
|
||||||
|
0x0469,HPSet,
|
||||||
|
0x046A,HPSet,
|
||||||
|
0x0471,RC300Summer2,
|
||||||
|
0x0472,RC300Summer2,
|
||||||
|
0x0473,RC300Summer2,
|
||||||
|
0x0474,RC300Summer2,
|
||||||
|
0x0475,RC300Summer2,
|
||||||
|
0x0476,RC300Summer2,
|
||||||
|
0x0477,RC300Summer2,
|
||||||
|
0x0478,RC300Summer2,
|
||||||
|
0x047B,HP2,
|
||||||
|
0x0484,HPSilentMode,fetched
|
||||||
|
0x0485,HpCooling,fetched
|
||||||
|
0x0486,HpInConfig,fetched
|
||||||
|
0x0488,HPValve,fetched
|
||||||
|
0x048A,HpPool,fetched
|
||||||
|
0x048B,HPPumps,fetched
|
||||||
|
0x048D,HpPower,fetched
|
||||||
|
0x048F,HpTemperatures,
|
||||||
|
0x0491,HPAdditionalHeater,fetched
|
||||||
|
0x0492,HpHeaterConfig,fetched
|
||||||
|
0x0494,UBAEnergySupplied,
|
||||||
|
0x0495,UBAInformation,
|
||||||
|
0x0499,HPDhwSettings,fetched
|
||||||
|
0x049C,HPSettings2,fetched
|
||||||
|
0x049D,HPSettings3,fetched
|
||||||
|
0x04A2,HpInput,fetched
|
||||||
|
0x04A5,HPFan,fetched
|
||||||
|
0x04A7,HPPowerLimit,fetched
|
||||||
|
0x04AA,HPPower,
|
||||||
|
0x04AE,HPEnergy,fetched
|
||||||
|
0x04AF,HPMeters,fetched
|
||||||
|
0x055C,VentilationSet,fetched
|
||||||
|
0x056B,VentilationMode,fetched
|
||||||
|
0x0583,VentilationMonitor,
|
||||||
|
0x0585,Blowerspeed,
|
||||||
|
0x0587,Bypass,
|
||||||
|
0x05BA,HpPoolStatus,fetched
|
||||||
|
0x05D9,Airquality,
|
||||||
|
0x0772,HIUSettings,
|
||||||
|
0x0779,HIUMonitor,
|
||||||
|
0x07A5,SM100wwCirc,fetched
|
||||||
|
0x07A6,SM100wwParam,fetched
|
||||||
|
0x07AA,SM100wwStatus,
|
||||||
|
0x07AB,SM100wwCommand,
|
||||||
|
0x07AC,SM100wwParam1,
|
||||||
|
0x07AD,SM100ValveStatus,
|
||||||
|
0x07AE,SM100wwKeepWarm,fetched
|
||||||
|
0x07D6,SM100wwTemperature,
|
||||||
|
0x07E0,SM100wwStatus2,fetched
|
||||||
|
0x0935,EM100SetMessage,fetched
|
||||||
|
0x0936,EM100OutMessage,
|
||||||
|
0x0937,EM100TempMessage,
|
||||||
|
0x0938,EM100InputMessage,
|
||||||
|
0x0939,EM100MonitorMessage,
|
||||||
|
0x093A,EM100ConfigMessage,
|
||||||
|
0x0998,HPSettings,fetched
|
||||||
|
0x0999,HPFunctionTest,fetched
|
||||||
|
0x099A,HPStarts,
|
||||||
|
0x099B,HPFlowTemp,
|
||||||
|
0x099C,HPComp,
|
||||||
|
0x09A0,HPTemperature,
|
||||||
|
|||||||
|
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"adapter": "react",
|
"adapter": "react",
|
||||||
"baseLocale": "pl",
|
"baseLocale": "pl",
|
||||||
"$schema": "https://unpkg.com/typesafe-i18n@5.26.2/schema/typesafe-i18n.json"
|
"$schema": "https://unpkg.com/typesafe-i18n@5.27.1/schema/typesafe-i18n.json"
|
||||||
}
|
}
|
||||||
@@ -1,9 +1,10 @@
|
|||||||
// @ts-check
|
// @ts-check
|
||||||
import eslint from '@eslint/js';
|
import eslint from '@eslint/js';
|
||||||
import prettierConfig from 'eslint-config-prettier';
|
import prettierConfig from 'eslint-config-prettier';
|
||||||
|
import { defineConfig } from 'eslint/config';
|
||||||
import tseslint from 'typescript-eslint';
|
import tseslint from 'typescript-eslint';
|
||||||
|
|
||||||
export default tseslint.config(
|
export default defineConfig(
|
||||||
eslint.configs.recommended,
|
eslint.configs.recommended,
|
||||||
...tseslint.configs.recommendedTypeChecked,
|
...tseslint.configs.recommendedTypeChecked,
|
||||||
prettierConfig,
|
prettierConfig,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "EMS-ESP",
|
"name": "EMS-ESP",
|
||||||
"version": "3.7.3",
|
"version": "3.8.0",
|
||||||
"description": "EMS-ESP WebUI",
|
"description": "EMS-ESP WebUI",
|
||||||
"homepage": "https://emsesp.org",
|
"homepage": "https://emsesp.org",
|
||||||
"author": "proddy, emsesp.org",
|
"author": "proddy, emsesp.org",
|
||||||
@@ -12,56 +12,59 @@
|
|||||||
"dev": "vite dev",
|
"dev": "vite dev",
|
||||||
"build": "vite build",
|
"build": "vite build",
|
||||||
"preview": "vite preview",
|
"preview": "vite preview",
|
||||||
"build-hosted": "typesafe-i18n && vite build --mode hosted",
|
"build-hosted": "typesafe-i18n --no-watch && vite build --mode hosted",
|
||||||
"preview-standalone": "typesafe-i18n --no-watch && vite build && concurrently -c \"auto\" \"pnpm:mock-rest\" \"vite preview\"",
|
|
||||||
"mock-rest": "bun --watch ../mock-api/restServer.ts",
|
"mock-rest": "bun --watch ../mock-api/restServer.ts",
|
||||||
"standalone": "concurrently -c \"auto\" \"typesafe-i18n\" \"pnpm:mock-rest\" \"vite\"",
|
"preview-standalone": "typesafe-i18n --no-watch && vite build && concurrently -c \"auto\" \"pnpm:mock-rest\" \"vite preview\"",
|
||||||
|
"standalone": "concurrently -c \"auto\" \"typesafe-i18n\" \"pnpm:mock-rest\" \"vite dev\"",
|
||||||
"typesafe-i18n": "typesafe-i18n --no-watch",
|
"typesafe-i18n": "typesafe-i18n --no-watch",
|
||||||
"webUI": "node progmem-generator.js",
|
"build_webUI": "typesafe-i18n --no-watch && vite build && node progmem-generator.js",
|
||||||
"format": "prettier -l -w '**/*.{ts,tsx,js,css,json,md}'",
|
"format": "prettier -l -w '**/*.{ts,tsx,js,css,json,md}'",
|
||||||
"lint": "eslint . --fix"
|
"lint": "eslint . --fix",
|
||||||
|
"standalone-devcontainer": "concurrently -c \"auto\" \"typesafe-i18n\" \"pnpm:mock-rest\" \"vite --host\""
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@alova/adapter-xhr": "2.2.1",
|
"@alova/adapter-xhr": "2.3.1",
|
||||||
"@emotion/react": "^11.14.0",
|
"@emotion/react": "^11.14.0",
|
||||||
"@emotion/styled": "^11.14.1",
|
"@emotion/styled": "^11.14.1",
|
||||||
"@mui/icons-material": "^7.3.4",
|
"@mui/icons-material": "^7.3.9",
|
||||||
"@mui/material": "^7.3.4",
|
"@mui/material": "^7.3.9",
|
||||||
|
"@preact/compat": "^18.3.2",
|
||||||
"@table-library/react-table-library": "4.1.15",
|
"@table-library/react-table-library": "4.1.15",
|
||||||
"alova": "3.3.4",
|
"alova": "^3.5.1",
|
||||||
"async-validator": "^4.2.5",
|
"async-validator": "^4.2.5",
|
||||||
|
"etag": "^1.8.1",
|
||||||
"formidable": "^3.5.4",
|
"formidable": "^3.5.4",
|
||||||
"jwt-decode": "^4.0.0",
|
"jwt-decode": "^4.0.0",
|
||||||
"magic-string": "^0.30.19",
|
"magic-string": "^0.30.21",
|
||||||
"mime-types": "^3.0.1",
|
"mime-types": "^3.0.2",
|
||||||
"preact": "^10.27.2",
|
"preact": "^10.29.0",
|
||||||
"react": "^19.2.0",
|
"react": "^19.2.4",
|
||||||
"react-dom": "^19.2.0",
|
"react-dom": "^19.2.4",
|
||||||
"react-icons": "^5.5.0",
|
"react-icons": "^5.6.0",
|
||||||
"react-router": "^7.9.4",
|
"react-router": "^7.13.1",
|
||||||
"react-toastify": "^11.0.5",
|
"react-toastify": "^11.0.5",
|
||||||
"typesafe-i18n": "^5.26.2",
|
"typesafe-i18n": "^5.27.1",
|
||||||
"typescript": "^5.9.3"
|
"typescript": "^5.9.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.28.4",
|
"@babel/core": "^7.29.0",
|
||||||
"@eslint/js": "^9.38.0",
|
"@eslint/js": "^10.0.1",
|
||||||
"@preact/compat": "^18.3.1",
|
"@preact/compat": "^18.3.2",
|
||||||
"@preact/preset-vite": "^2.10.2",
|
"@preact/preset-vite": "^2.10.5",
|
||||||
"@trivago/prettier-plugin-sort-imports": "^5.2.2",
|
"@trivago/prettier-plugin-sort-imports": "^6.0.2",
|
||||||
"@types/node": "^24.9.1",
|
"@types/node": "^25.5.0",
|
||||||
"@types/react": "^19.2.2",
|
"@types/react": "^19.2.14",
|
||||||
"@types/react-dom": "^19.2.2",
|
"@types/react-dom": "^19.2.3",
|
||||||
|
"axe-core": "^4.11.1",
|
||||||
"concurrently": "^9.2.1",
|
"concurrently": "^9.2.1",
|
||||||
"eslint": "^9.38.0",
|
"eslint": "^10.1.0",
|
||||||
"eslint-config-prettier": "^10.1.8",
|
"eslint-config-prettier": "^10.1.8",
|
||||||
"prettier": "^3.6.2",
|
"prettier": "^3.8.1",
|
||||||
"rollup-plugin-visualizer": "^6.0.5",
|
"rollup-plugin-visualizer": "^7.0.1",
|
||||||
"terser": "^5.44.0",
|
"terser": "^5.46.1",
|
||||||
"typescript-eslint": "^8.46.2",
|
"typescript-eslint": "^8.57.1",
|
||||||
"vite": "^7.1.11",
|
"vite": "^8.0.1",
|
||||||
"vite-plugin-imagemin": "^0.6.1",
|
"vite-plugin-imagemin": "^0.6.1"
|
||||||
"vite-tsconfig-paths": "^5.1.4"
|
|
||||||
},
|
},
|
||||||
"packageManager": "pnpm@10.19.0+sha512.c9fc7236e92adf5c8af42fd5bf1612df99c2ceb62f27047032f4720b33f8eacdde311865e91c411f2774f618d82f320808ecb51718bfa82c060c4ba7c76a32b8"
|
"packageManager": "pnpm@10.32.1+sha512.a706938f0e89ac1456b6563eab4edf1d1faf3368d1191fc5c59790e96dc918e4456ab2e67d613de1043d2e8c81f87303e6b40d4ffeca9df15ef1ad567348f2be"
|
||||||
}
|
}
|
||||||
|
|||||||
2513
interface/pnpm-lock.yaml
generated
2513
interface/pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@@ -1,10 +1,9 @@
|
|||||||
import crypto from 'crypto';
|
import etag from 'etag';
|
||||||
import {
|
import {
|
||||||
createWriteStream,
|
createWriteStream,
|
||||||
existsSync,
|
existsSync,
|
||||||
readFileSync,
|
readFileSync,
|
||||||
readdirSync,
|
readdirSync,
|
||||||
statSync,
|
|
||||||
unlinkSync
|
unlinkSync
|
||||||
} from 'fs';
|
} from 'fs';
|
||||||
import mime from 'mime-types';
|
import mime from 'mime-types';
|
||||||
@@ -36,7 +35,7 @@ const generateWWWClass =
|
|||||||
class WWWData {
|
class WWWData {
|
||||||
${INDENT}public:
|
${INDENT}public:
|
||||||
${INDENT.repeat(2)}static void registerRoutes(RouteRegistrationHandler handler) {
|
${INDENT.repeat(2)}static void registerRoutes(RouteRegistrationHandler handler) {
|
||||||
${fileInfo.map((f) => `${INDENT.repeat(3)}handler("${f.uri}", "${f.mimeType}", ${f.variable}, ${f.size}, "${f.hash}");`).join('\n')}
|
${fileInfo.map((f) => `${INDENT.repeat(3)}handler("${f.uri}", "${f.mimeType}", ${f.variable}, ${f.size}, ${f.hash});`).join('\n')}
|
||||||
${INDENT.repeat(2)}}
|
${INDENT.repeat(2)}}
|
||||||
};
|
};
|
||||||
`;
|
`;
|
||||||
@@ -71,7 +70,8 @@ const writeFile = (relativeFilePath, buffer) => {
|
|||||||
writeStream.write(`const uint8_t ${variable}[] = {`);
|
writeStream.write(`const uint8_t ${variable}[] = {`);
|
||||||
|
|
||||||
const zipBuffer = zlib.gzipSync(buffer, { level: 9 });
|
const zipBuffer = zlib.gzipSync(buffer, { level: 9 });
|
||||||
const hash = crypto.createHash('sha256').update(zipBuffer).digest('hex');
|
// const hash = crypto.createHash('sha256').update(zipBuffer).digest('hex');
|
||||||
|
const hash = etag(zipBuffer); // use smaller md5 instead of sha256
|
||||||
|
|
||||||
zipBuffer.forEach((b) => {
|
zipBuffer.forEach((b) => {
|
||||||
if (!(size % bytesPerLine)) {
|
if (!(size % bytesPerLine)) {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { memo, useCallback, useEffect, useState } from 'react';
|
||||||
import { ToastContainer, Zoom } from 'react-toastify';
|
import { ToastContainer, Zoom } from 'react-toastify';
|
||||||
|
|
||||||
import AppRouting from 'AppRouting';
|
import AppRouting from 'AppRouting';
|
||||||
@@ -8,7 +8,7 @@ import type { Locales } from 'i18n/i18n-types';
|
|||||||
import { loadLocaleAsync } from 'i18n/i18n-util.async';
|
import { loadLocaleAsync } from 'i18n/i18n-util.async';
|
||||||
import { detectLocale, navigatorDetector } from 'typesafe-i18n/detectors';
|
import { detectLocale, navigatorDetector } from 'typesafe-i18n/detectors';
|
||||||
|
|
||||||
const availableLocales = [
|
const AVAILABLE_LOCALES = [
|
||||||
'de',
|
'de',
|
||||||
'en',
|
'en',
|
||||||
'it',
|
'it',
|
||||||
@@ -20,47 +20,56 @@ const availableLocales = [
|
|||||||
'sv',
|
'sv',
|
||||||
'tr',
|
'tr',
|
||||||
'cz'
|
'cz'
|
||||||
];
|
] as Locales[];
|
||||||
|
|
||||||
const App = () => {
|
// Static toast configuration - no need to recreate on every render
|
||||||
|
const TOAST_CONTAINER_PROPS = {
|
||||||
|
position: 'bottom-left' as const,
|
||||||
|
autoClose: 3000,
|
||||||
|
hideProgressBar: false,
|
||||||
|
newestOnTop: false,
|
||||||
|
closeOnClick: true,
|
||||||
|
rtl: false,
|
||||||
|
pauseOnFocusLoss: true,
|
||||||
|
draggable: false,
|
||||||
|
pauseOnHover: false,
|
||||||
|
transition: Zoom,
|
||||||
|
closeButton: false,
|
||||||
|
theme: 'dark' as const,
|
||||||
|
toastStyle: {
|
||||||
|
border: '1px solid #177ac9',
|
||||||
|
width: 'fit-content'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const App = memo(() => {
|
||||||
const [wasLoaded, setWasLoaded] = useState(false);
|
const [wasLoaded, setWasLoaded] = useState(false);
|
||||||
const [locale, setLocale] = useState<Locales>('en');
|
const [locale, setLocale] = useState<Locales>('en');
|
||||||
|
|
||||||
useEffect(() => {
|
// Memoize locale initialization to prevent unnecessary re-runs
|
||||||
// determine locale, take from session if set other default to browser language
|
const initializeLocale = useCallback(async () => {
|
||||||
const browserLocale = detectLocale('en', availableLocales, navigatorDetector);
|
const browserLocale = detectLocale('en', AVAILABLE_LOCALES, navigatorDetector);
|
||||||
const newLocale = (localStorage.getItem('lang') || browserLocale) as Locales;
|
const newLocale = (localStorage.getItem('lang') || browserLocale) as Locales;
|
||||||
localStorage.setItem('lang', newLocale);
|
localStorage.setItem('lang', newLocale);
|
||||||
setLocale(newLocale);
|
setLocale(newLocale);
|
||||||
void loadLocaleAsync(newLocale).then(() => setWasLoaded(true));
|
await loadLocaleAsync(newLocale);
|
||||||
|
setWasLoaded(true);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
void initializeLocale();
|
||||||
|
}, [initializeLocale]);
|
||||||
|
|
||||||
if (!wasLoaded) return null;
|
if (!wasLoaded) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<TypesafeI18n locale={locale}>
|
<TypesafeI18n locale={locale}>
|
||||||
<CustomTheme>
|
<CustomTheme>
|
||||||
<AppRouting />
|
<AppRouting />
|
||||||
<ToastContainer
|
<ToastContainer {...TOAST_CONTAINER_PROPS} />
|
||||||
position="bottom-left"
|
|
||||||
autoClose={3000}
|
|
||||||
hideProgressBar={false}
|
|
||||||
newestOnTop={false}
|
|
||||||
closeOnClick
|
|
||||||
rtl={false}
|
|
||||||
pauseOnFocusLoss
|
|
||||||
draggable={false}
|
|
||||||
pauseOnHover={false}
|
|
||||||
transition={Zoom}
|
|
||||||
closeButton={false}
|
|
||||||
theme="dark"
|
|
||||||
toastStyle={{
|
|
||||||
border: '1px solid #177ac9'
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</CustomTheme>
|
</CustomTheme>
|
||||||
</TypesafeI18n>
|
</TypesafeI18n>
|
||||||
);
|
);
|
||||||
};
|
});
|
||||||
|
|
||||||
export default App;
|
export default App;
|
||||||
|
|||||||
@@ -1,32 +1,51 @@
|
|||||||
import { useContext, useEffect } from 'react';
|
import { type FC, Suspense, lazy, memo, useContext, useEffect, useRef } from 'react';
|
||||||
import { Navigate, Route, Routes } from 'react-router';
|
import { Navigate, Route, Routes } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import AuthenticatedRouting from 'AuthenticatedRouting';
|
import {
|
||||||
import SignIn from 'SignIn';
|
LoadingSpinner,
|
||||||
import { RequireAuthenticated, RequireUnauthenticated } from 'components';
|
RequireAuthenticated,
|
||||||
|
RequireUnauthenticated
|
||||||
|
} from 'components';
|
||||||
import { Authentication, AuthenticationContext } from 'contexts/authentication';
|
import { Authentication, AuthenticationContext } from 'contexts/authentication';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
|
|
||||||
|
// Lazy load route components for better code splitting
|
||||||
|
const SignIn = lazy(() => import('SignIn'));
|
||||||
|
const AuthenticatedRouting = lazy(() => import('AuthenticatedRouting'));
|
||||||
|
|
||||||
interface SecurityRedirectProps {
|
interface SecurityRedirectProps {
|
||||||
message: string;
|
readonly message: string;
|
||||||
signOut?: boolean;
|
readonly signOut?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const RootRedirect = ({ message, signOut }: SecurityRedirectProps) => {
|
const RootRedirect: FC<SecurityRedirectProps> = memo(
|
||||||
const authenticationContext = useContext(AuthenticationContext);
|
({ message, signOut = false }) => {
|
||||||
useEffect(() => {
|
const { signOut: contextSignOut } = useContext(AuthenticationContext);
|
||||||
signOut && authenticationContext.signOut(false);
|
const hasShownToast = useRef(false);
|
||||||
toast.success(message);
|
|
||||||
}, [message, signOut, authenticationContext]);
|
|
||||||
return <Navigate to="/" />;
|
|
||||||
};
|
|
||||||
|
|
||||||
const AppRouting = () => {
|
useEffect(() => {
|
||||||
|
// Prevent duplicate toasts on strict mode or re-renders
|
||||||
|
if (!hasShownToast.current) {
|
||||||
|
hasShownToast.current = true;
|
||||||
|
if (signOut) {
|
||||||
|
contextSignOut(false);
|
||||||
|
}
|
||||||
|
toast.success(message);
|
||||||
|
}
|
||||||
|
// Only run once on mount - using ref to track execution
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return <Navigate to="/" replace />;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const AppRouting: FC = memo(() => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Authentication>
|
<Authentication>
|
||||||
|
<Suspense fallback={<LoadingSpinner />}>
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route
|
<Route
|
||||||
path="/unauthorized"
|
path="/unauthorized"
|
||||||
@@ -53,8 +72,9 @@ const AppRouting = () => {
|
|||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</Routes>
|
</Routes>
|
||||||
|
</Suspense>
|
||||||
</Authentication>
|
</Authentication>
|
||||||
);
|
);
|
||||||
};
|
});
|
||||||
|
|
||||||
export default AppRouting;
|
export default AppRouting;
|
||||||
|
|||||||
@@ -1,43 +1,50 @@
|
|||||||
import { useContext } from 'react';
|
import { Suspense, lazy, memo, useContext } from 'react';
|
||||||
import { Navigate, Route, Routes } from 'react-router';
|
import { Navigate, Route, Routes } from 'react-router';
|
||||||
|
|
||||||
import CustomEntities from 'app/main/CustomEntities';
|
import { Layout, LoadingSpinner } from 'components';
|
||||||
import Customizations from 'app/main/Customizations';
|
|
||||||
import Dashboard from 'app/main/Dashboard';
|
|
||||||
import Devices from 'app/main/Devices';
|
|
||||||
import Help from 'app/main/Help';
|
|
||||||
import Modules from 'app/main/Modules';
|
|
||||||
import Scheduler from 'app/main/Scheduler';
|
|
||||||
import Sensors from 'app/main/Sensors';
|
|
||||||
import APSettings from 'app/settings/APSettings';
|
|
||||||
import ApplicationSettings from 'app/settings/ApplicationSettings';
|
|
||||||
import DownloadUpload from 'app/settings/DownloadUpload';
|
|
||||||
import MqttSettings from 'app/settings/MqttSettings';
|
|
||||||
import NTPSettings from 'app/settings/NTPSettings';
|
|
||||||
import Settings from 'app/settings/Settings';
|
|
||||||
import Network from 'app/settings/network/Network';
|
|
||||||
import Security from 'app/settings/security/Security';
|
|
||||||
import APStatus from 'app/status/APStatus';
|
|
||||||
import Activity from 'app/status/Activity';
|
|
||||||
import HardwareStatus from 'app/status/HardwareStatus';
|
|
||||||
import MqttStatus from 'app/status/MqttStatus';
|
|
||||||
import NTPStatus from 'app/status/NTPStatus';
|
|
||||||
import NetworkStatus from 'app/status/NetworkStatus';
|
|
||||||
import Status from 'app/status/Status';
|
|
||||||
import SystemLog from 'app/status/SystemLog';
|
|
||||||
import Version from 'app/status/Version';
|
|
||||||
import { Layout } from 'components';
|
|
||||||
import { AuthenticatedContext } from 'contexts/authentication';
|
import { AuthenticatedContext } from 'contexts/authentication';
|
||||||
|
|
||||||
const AuthenticatedRouting = () => {
|
// Lazy load all route components for better code splitting
|
||||||
|
const Dashboard = lazy(() => import('app/main/Dashboard'));
|
||||||
|
const Devices = lazy(() => import('app/main/Devices'));
|
||||||
|
const Sensors = lazy(() => import('app/main/Sensors'));
|
||||||
|
const Help = lazy(() => import('app/main/Help'));
|
||||||
|
const Customizations = lazy(() => import('app/main/Customizations'));
|
||||||
|
const Scheduler = lazy(() => import('app/main/Scheduler'));
|
||||||
|
const CustomEntities = lazy(() => import('app/main/CustomEntities'));
|
||||||
|
const Modules = lazy(() => import('app/main/Modules'));
|
||||||
|
const UserProfile = lazy(() => import('app/main/UserProfile'));
|
||||||
|
|
||||||
|
const Status = lazy(() => import('app/status/Status'));
|
||||||
|
const HardwareStatus = lazy(() => import('app/status/HardwareStatus'));
|
||||||
|
const Activity = lazy(() => import('app/status/Activity'));
|
||||||
|
const SystemLog = lazy(() => import('app/status/SystemLog'));
|
||||||
|
const MqttStatus = lazy(() => import('app/status/MqttStatus'));
|
||||||
|
const NTPStatus = lazy(() => import('app/status/NTPStatus'));
|
||||||
|
const APStatus = lazy(() => import('app/status/APStatus'));
|
||||||
|
const NetworkStatus = lazy(() => import('app/status/NetworkStatus'));
|
||||||
|
const Version = lazy(() => import('app/status/Version'));
|
||||||
|
|
||||||
|
const Settings = lazy(() => import('app/settings/Settings'));
|
||||||
|
const ApplicationSettings = lazy(() => import('app/settings/ApplicationSettings'));
|
||||||
|
const MqttSettings = lazy(() => import('app/settings/MqttSettings'));
|
||||||
|
const NTPSettings = lazy(() => import('app/settings/NTPSettings'));
|
||||||
|
const APSettings = lazy(() => import('app/settings/APSettings'));
|
||||||
|
const DownloadUpload = lazy(() => import('app/settings/DownloadUpload'));
|
||||||
|
const Network = lazy(() => import('app/settings/network/Network'));
|
||||||
|
const Security = lazy(() => import('app/settings/security/Security'));
|
||||||
|
|
||||||
|
const AuthenticatedRouting = memo(() => {
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
return (
|
return (
|
||||||
<Layout>
|
<Layout>
|
||||||
|
<Suspense fallback={<LoadingSpinner />}>
|
||||||
<Routes>
|
<Routes>
|
||||||
<Route path="/dashboard/*" element={<Dashboard />} />
|
<Route path="/dashboard/*" element={<Dashboard />} />
|
||||||
<Route path="/devices/*" element={<Devices />} />
|
<Route path="/devices/*" element={<Devices />} />
|
||||||
<Route path="/sensors/*" element={<Sensors />} />
|
<Route path="/sensors/*" element={<Sensors />} />
|
||||||
<Route path="/help/*" element={<Help />} />
|
<Route path="/help/*" element={<Help />} />
|
||||||
|
<Route path="/user/*" element={<UserProfile />} />
|
||||||
|
|
||||||
<Route path="/status/*" element={<Status />} />
|
<Route path="/status/*" element={<Status />} />
|
||||||
<Route path="/status/hardwarestatus/*" element={<HardwareStatus />} />
|
<Route path="/status/hardwarestatus/*" element={<HardwareStatus />} />
|
||||||
@@ -52,7 +59,10 @@ const AuthenticatedRouting = () => {
|
|||||||
{me.admin && (
|
{me.admin && (
|
||||||
<>
|
<>
|
||||||
<Route path="/settings" element={<Settings />} />
|
<Route path="/settings" element={<Settings />} />
|
||||||
<Route path="/settings/application" element={<ApplicationSettings />} />
|
<Route
|
||||||
|
path="/settings/application"
|
||||||
|
element={<ApplicationSettings />}
|
||||||
|
/>
|
||||||
<Route path="/settings/mqtt" element={<MqttSettings />} />
|
<Route path="/settings/mqtt" element={<MqttSettings />} />
|
||||||
<Route path="/settings/ntp" element={<NTPSettings />} />
|
<Route path="/settings/ntp" element={<NTPSettings />} />
|
||||||
<Route path="/settings/ap" element={<APSettings />} />
|
<Route path="/settings/ap" element={<APSettings />} />
|
||||||
@@ -70,8 +80,9 @@ const AuthenticatedRouting = () => {
|
|||||||
|
|
||||||
<Route path="/*" element={<Navigate to="/" />} />
|
<Route path="/*" element={<Navigate to="/" />} />
|
||||||
</Routes>
|
</Routes>
|
||||||
|
</Suspense>
|
||||||
</Layout>
|
</Layout>
|
||||||
);
|
);
|
||||||
};
|
});
|
||||||
|
|
||||||
export default AuthenticatedRouting;
|
export default AuthenticatedRouting;
|
||||||
|
|||||||
@@ -1,6 +1,12 @@
|
|||||||
|
import { memo } from 'react';
|
||||||
import type { FC } from 'react';
|
import type { FC } from 'react';
|
||||||
|
|
||||||
import { CssBaseline, ThemeProvider, responsiveFontSizes } from '@mui/material';
|
import {
|
||||||
|
CssBaseline,
|
||||||
|
ThemeProvider,
|
||||||
|
responsiveFontSizes,
|
||||||
|
tooltipClasses
|
||||||
|
} from '@mui/material';
|
||||||
import { createTheme } from '@mui/material/styles';
|
import { createTheme } from '@mui/material/styles';
|
||||||
|
|
||||||
import type { RequiredChildrenProps } from 'utils';
|
import type { RequiredChildrenProps } from 'utils';
|
||||||
@@ -10,9 +16,9 @@ export const dialogStyle = {
|
|||||||
borderRadius: '8px',
|
borderRadius: '8px',
|
||||||
borderColor: '#565656',
|
borderColor: '#565656',
|
||||||
borderStyle: 'solid',
|
borderStyle: 'solid',
|
||||||
borderWidth: '1px'
|
borderWidth: '2px'
|
||||||
}
|
}
|
||||||
};
|
} as const;
|
||||||
|
|
||||||
const theme = responsiveFontSizes(
|
const theme = responsiveFontSizes(
|
||||||
createTheme({
|
createTheme({
|
||||||
@@ -30,15 +36,45 @@ const theme = responsiveFontSizes(
|
|||||||
text: {
|
text: {
|
||||||
disabled: '#eee' // white
|
disabled: '#eee' // white
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
components: {
|
||||||
|
MuiListItemText: {
|
||||||
|
styleOverrides: {
|
||||||
|
primary: {
|
||||||
|
fontSize: 14
|
||||||
|
},
|
||||||
|
secondary: {
|
||||||
|
color: '#9e9e9e' // grey[500]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
MuiTooltip: {
|
||||||
|
defaultProps: {
|
||||||
|
placement: 'top',
|
||||||
|
arrow: true
|
||||||
|
},
|
||||||
|
styleOverrides: {
|
||||||
|
tooltip: {
|
||||||
|
padding: '4px 8px',
|
||||||
|
fontSize: 10,
|
||||||
|
color: 'rgba(0, 0, 0, 0.87)',
|
||||||
|
backgroundColor: '#4caf50', // MUI success.main default color
|
||||||
|
boxShadow: '0px 2px 8px rgba(0, 0, 0, 0.15)',
|
||||||
|
[`& .${tooltipClasses.arrow}`]: {
|
||||||
|
color: '#4caf50'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
const CustomTheme: FC<RequiredChildrenProps> = ({ children }) => (
|
const CustomTheme: FC<RequiredChildrenProps> = memo(({ children }) => (
|
||||||
<ThemeProvider theme={theme}>
|
<ThemeProvider theme={theme}>
|
||||||
<CssBaseline />
|
<CssBaseline />
|
||||||
{children}
|
{children}
|
||||||
</ThemeProvider>
|
</ThemeProvider>
|
||||||
);
|
));
|
||||||
|
|
||||||
export default CustomTheme;
|
export default CustomTheme;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useContext, useState } from 'react';
|
import { memo, useCallback, useContext, useEffect, useMemo, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import ForwardIcon from '@mui/icons-material/Forward';
|
import ForwardIcon from '@mui/icons-material/Forward';
|
||||||
@@ -19,7 +19,7 @@ import type { SignInRequest } from 'types';
|
|||||||
import { onEnterCallback, updateValue } from 'utils';
|
import { onEnterCallback, updateValue } from 'utils';
|
||||||
import { SIGN_IN_REQUEST_VALIDATOR, validate } from 'validators';
|
import { SIGN_IN_REQUEST_VALIDATOR, validate } from 'validators';
|
||||||
|
|
||||||
const SignIn = () => {
|
const SignIn = memo(() => {
|
||||||
const authenticationContext = useContext(AuthenticationContext);
|
const authenticationContext = useContext(AuthenticationContext);
|
||||||
|
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
@@ -42,9 +42,18 @@ const SignIn = () => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const updateLoginRequestValue = updateValue(setSignInRequest);
|
// Memoize callback to prevent recreation on every render
|
||||||
|
const updateLoginRequestValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValue((updater) =>
|
||||||
|
setSignInRequest(
|
||||||
|
updater as unknown as (prevState: SignInRequest) => SignInRequest
|
||||||
|
)
|
||||||
|
),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
const signIn = async () => {
|
const signIn = useCallback(async () => {
|
||||||
await callSignIn(signInRequest).catch((event: Error) => {
|
await callSignIn(signInRequest).catch((event: Error) => {
|
||||||
if (event.message === 'Unauthorized') {
|
if (event.message === 'Unauthorized') {
|
||||||
toast.warning(LL.INVALID_LOGIN());
|
toast.warning(LL.INVALID_LOGIN());
|
||||||
@@ -53,9 +62,9 @@ const SignIn = () => {
|
|||||||
}
|
}
|
||||||
setProcessing(false);
|
setProcessing(false);
|
||||||
});
|
});
|
||||||
};
|
}, [callSignIn, signInRequest, LL]);
|
||||||
|
|
||||||
const validateAndSignIn = async () => {
|
const validateAndSignIn = useCallback(async () => {
|
||||||
setProcessing(true);
|
setProcessing(true);
|
||||||
SIGN_IN_REQUEST_VALIDATOR.messages({
|
SIGN_IN_REQUEST_VALIDATOR.messages({
|
||||||
required: LL.IS_REQUIRED('%s')
|
required: LL.IS_REQUIRED('%s')
|
||||||
@@ -67,9 +76,19 @@ const SignIn = () => {
|
|||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
setProcessing(false);
|
setProcessing(false);
|
||||||
}
|
}
|
||||||
};
|
}, [signInRequest, signIn, LL]);
|
||||||
|
|
||||||
const submitOnEnter = onEnterCallback(signIn);
|
// Memoize callback to prevent recreation on every render
|
||||||
|
const submitOnEnter = useMemo(() => onEnterCallback(signIn), [signIn]);
|
||||||
|
|
||||||
|
// get rid of scrollbar
|
||||||
|
useEffect(() => {
|
||||||
|
const originalOverflow = document.body.style.overflow;
|
||||||
|
document.body.style.overflow = 'hidden';
|
||||||
|
return () => {
|
||||||
|
document.body.style.overflow = originalOverflow;
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box
|
<Box
|
||||||
@@ -92,23 +111,27 @@ const SignIn = () => {
|
|||||||
width: '100%'
|
width: '100%'
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Typography variant="h4">{PROJECT_NAME}</Typography>
|
<Typography mb={1} variant="h4">
|
||||||
|
{PROJECT_NAME}
|
||||||
|
</Typography>
|
||||||
<LanguageSelector />
|
<LanguageSelector />
|
||||||
|
<Box
|
||||||
<Box display="flex" flexDirection="column" alignItems="center">
|
mt={1}
|
||||||
|
display="flex"
|
||||||
|
flexDirection="column"
|
||||||
|
gap={1}
|
||||||
|
alignItems="center"
|
||||||
|
>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors || {}}
|
||||||
disabled={processing}
|
disabled={processing}
|
||||||
sx={{
|
sx={{
|
||||||
width: 240
|
width: '32ch'
|
||||||
}}
|
}}
|
||||||
name="username"
|
name="username"
|
||||||
label={LL.USERNAME(0)}
|
label={LL.USERNAME(0)}
|
||||||
value={signInRequest.username}
|
value={signInRequest.username}
|
||||||
onChange={updateLoginRequestValue}
|
onChange={updateLoginRequestValue}
|
||||||
margin="normal"
|
|
||||||
variant="outlined"
|
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
autoCapitalize: 'none',
|
autoCapitalize: 'none',
|
||||||
@@ -120,14 +143,13 @@ const SignIn = () => {
|
|||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors || {}}
|
||||||
disabled={processing}
|
disabled={processing}
|
||||||
sx={{
|
sx={{
|
||||||
width: 240
|
width: '32ch'
|
||||||
}}
|
}}
|
||||||
name="password"
|
name="password"
|
||||||
label={LL.PASSWORD()}
|
label={LL.PASSWORD()}
|
||||||
value={signInRequest.password}
|
value={signInRequest.password}
|
||||||
onChange={updateLoginRequestValue}
|
onChange={updateLoginRequestValue}
|
||||||
onKeyDown={submitOnEnter}
|
onKeyDown={submitOnEnter}
|
||||||
variant="outlined"
|
|
||||||
/>
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
@@ -144,6 +166,6 @@ const SignIn = () => {
|
|||||||
</Paper>
|
</Paper>
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
};
|
});
|
||||||
|
|
||||||
export default SignIn;
|
export default SignIn;
|
||||||
|
|||||||
@@ -20,19 +20,18 @@ import type {
|
|||||||
WriteTemperatureSensor
|
WriteTemperatureSensor
|
||||||
} from '../app/main/types';
|
} from '../app/main/types';
|
||||||
|
|
||||||
|
const MSGPACK_CONFIG = { responseType: 'arraybuffer' as const };
|
||||||
|
|
||||||
// Dashboard
|
// Dashboard
|
||||||
export const readDashboard = () =>
|
export const readDashboard = () =>
|
||||||
alovaInstance.Get<DashboardData>('/rest/dashboardData', {
|
alovaInstance.Get<DashboardData>('/rest/dashboardData', MSGPACK_CONFIG);
|
||||||
responseType: 'arraybuffer' // uses msgpack
|
|
||||||
});
|
|
||||||
|
|
||||||
// Devices
|
// Devices
|
||||||
export const readCoreData = () => alovaInstance.Get<CoreData>(`/rest/coreData`);
|
export const readCoreData = () => alovaInstance.Get<CoreData>('/rest/coreData');
|
||||||
export const readDeviceData = (id: number) =>
|
export const readDeviceData = (id: number) =>
|
||||||
alovaInstance.Get<DeviceData>('/rest/deviceData', {
|
alovaInstance.Get<DeviceData>('/rest/deviceData', {
|
||||||
// alovaInstance.Get<DeviceData>(`/rest/deviceData/${id}`, {
|
|
||||||
params: { id },
|
params: { id },
|
||||||
responseType: 'arraybuffer' // uses msgpack
|
...MSGPACK_CONFIG
|
||||||
});
|
});
|
||||||
export const writeDeviceValue = (data: { id: number; c: string; v: unknown }) =>
|
export const writeDeviceValue = (data: { id: number; c: string; v: unknown }) =>
|
||||||
alovaInstance.Post('/rest/writeDeviceValue', data);
|
alovaInstance.Post('/rest/writeDeviceValue', data);
|
||||||
@@ -66,13 +65,13 @@ export const callAction = (action: Action) =>
|
|||||||
|
|
||||||
// SettingsCustomization
|
// SettingsCustomization
|
||||||
export const readDeviceEntities = (id: number) =>
|
export const readDeviceEntities = (id: number) =>
|
||||||
// alovaInstance.Get<DeviceEntity[]>(`/rest/deviceEntities/${id}`, {
|
alovaInstance.Get<DeviceEntity[]>('/rest/deviceEntities', {
|
||||||
alovaInstance.Get<DeviceEntity[]>(`/rest/deviceEntities`, {
|
|
||||||
params: { id },
|
params: { id },
|
||||||
responseType: 'arraybuffer',
|
...MSGPACK_CONFIG,
|
||||||
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
||||||
transform(data) {
|
transform(data) {
|
||||||
return (data as DeviceEntity[]).map((de: DeviceEntity) => ({
|
const entities = data as DeviceEntity[];
|
||||||
|
return entities.map((de) => ({
|
||||||
...de,
|
...de,
|
||||||
o_m: de.m,
|
o_m: de.m,
|
||||||
o_cn: de.cn,
|
o_cn: de.cn,
|
||||||
@@ -95,7 +94,8 @@ export const readSchedule = () =>
|
|||||||
alovaInstance.Get<ScheduleItem[]>('/rest/schedule', {
|
alovaInstance.Get<ScheduleItem[]>('/rest/schedule', {
|
||||||
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
||||||
transform(data) {
|
transform(data) {
|
||||||
return (data as Schedule).schedule.map((si: ScheduleItem) => ({
|
const schedule = (data as Schedule).schedule;
|
||||||
|
return schedule.map((si) => ({
|
||||||
...si,
|
...si,
|
||||||
o_id: si.id,
|
o_id: si.id,
|
||||||
o_active: si.active,
|
o_active: si.active,
|
||||||
@@ -115,7 +115,8 @@ export const writeSchedule = (data: Schedule) =>
|
|||||||
export const readModules = () =>
|
export const readModules = () =>
|
||||||
alovaInstance.Get<ModuleItem[]>('/rest/modules', {
|
alovaInstance.Get<ModuleItem[]>('/rest/modules', {
|
||||||
transform(data) {
|
transform(data) {
|
||||||
return (data as Modules).modules.map((mi: ModuleItem) => ({
|
const modules = (data as Modules).modules;
|
||||||
|
return modules.map((mi) => ({
|
||||||
...mi,
|
...mi,
|
||||||
o_enabled: mi.enabled,
|
o_enabled: mi.enabled,
|
||||||
o_license: mi.license
|
o_license: mi.license
|
||||||
@@ -133,7 +134,8 @@ export const readCustomEntities = () =>
|
|||||||
alovaInstance.Get<EntityItem[]>('/rest/customEntities', {
|
alovaInstance.Get<EntityItem[]>('/rest/customEntities', {
|
||||||
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
||||||
transform(data) {
|
transform(data) {
|
||||||
return (data as Entities).entities.map((ei: EntityItem) => ({
|
const entities = (data as Entities).entities;
|
||||||
|
return entities.map((ei) => ({
|
||||||
...ei,
|
...ei,
|
||||||
o_id: ei.id,
|
o_id: ei.id,
|
||||||
o_ram: ei.ram,
|
o_ram: ei.ram,
|
||||||
|
|||||||
@@ -4,55 +4,57 @@ import ReactHook from 'alova/react';
|
|||||||
|
|
||||||
import { unpack } from './unpack';
|
import { unpack } from './unpack';
|
||||||
|
|
||||||
export const ACCESS_TOKEN = 'access_token';
|
export const ACCESS_TOKEN = 'access_token' as const;
|
||||||
|
|
||||||
|
// Cached token to avoid repeated localStorage access
|
||||||
|
let cachedToken: string | null = null;
|
||||||
|
|
||||||
|
const getAccessToken = (): string | null => {
|
||||||
|
if (cachedToken === null) {
|
||||||
|
cachedToken = localStorage.getItem(ACCESS_TOKEN);
|
||||||
|
}
|
||||||
|
return cachedToken;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Clear token cache when needed (e.g., on logout)
|
||||||
|
export const clearTokenCache = (): void => {
|
||||||
|
cachedToken = null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleResponse = async (response: AlovaXHRResponse) => {
|
||||||
|
// Handle various HTTP status codes
|
||||||
|
if (response.status === 205) {
|
||||||
|
throw new Error('Reboot required');
|
||||||
|
}
|
||||||
|
if (response.status === 400) {
|
||||||
|
throw new Error('Request Failed');
|
||||||
|
}
|
||||||
|
if (response.status >= 400) {
|
||||||
|
throw new Error(response.statusText);
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = (await response.data) as ArrayBuffer;
|
||||||
|
|
||||||
|
// Unpack MessagePack data if ArrayBuffer
|
||||||
|
if (data instanceof ArrayBuffer) {
|
||||||
|
return unpack(data) as ArrayBuffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
return data;
|
||||||
|
};
|
||||||
|
|
||||||
export const alovaInstance = createAlova({
|
export const alovaInstance = createAlova({
|
||||||
statesHook: ReactHook,
|
statesHook: ReactHook,
|
||||||
// timeout: 3000, // 3 seconds before throwing a timeout error, default is 0 = none
|
|
||||||
cacheFor: null, // disable cache
|
cacheFor: null, // disable cache
|
||||||
// cacheFor: {
|
|
||||||
// GET: {
|
|
||||||
// mode: 'memory',
|
|
||||||
// expire: 60 * 10 * 1000 // 60 seconds in cache
|
|
||||||
// }
|
|
||||||
// },
|
|
||||||
requestAdapter: xhrRequestAdapter(),
|
requestAdapter: xhrRequestAdapter(),
|
||||||
beforeRequest(method) {
|
beforeRequest(method) {
|
||||||
if (localStorage.getItem(ACCESS_TOKEN)) {
|
const token = getAccessToken();
|
||||||
method.config.headers.Authorization =
|
if (token) {
|
||||||
'Bearer ' + localStorage.getItem(ACCESS_TOKEN);
|
method.config.headers.Authorization = `Bearer ${token}`;
|
||||||
}
|
}
|
||||||
// for simulating very slow networks
|
|
||||||
// return new Promise((resolve) => {
|
|
||||||
// const random = 3000 + Math.random() * 2000;
|
|
||||||
// setTimeout(resolve, Math.floor(random));
|
|
||||||
// });
|
|
||||||
},
|
},
|
||||||
|
|
||||||
responded: {
|
responded: {
|
||||||
onSuccess: async (response: AlovaXHRResponse) => {
|
onSuccess: handleResponse
|
||||||
// if (response.status === 202) {
|
|
||||||
// throw new Error('Wait'); // wifi scan in progress
|
|
||||||
// } else
|
|
||||||
if (response.status === 205) {
|
|
||||||
throw new Error('Reboot required');
|
|
||||||
} else if (response.status === 400) {
|
|
||||||
throw new Error('Request Failed');
|
|
||||||
} else if (response.status >= 400) {
|
|
||||||
throw new Error(response.statusText);
|
|
||||||
}
|
|
||||||
const data: ArrayBuffer = (await response.data) as ArrayBuffer;
|
|
||||||
if (response.data instanceof ArrayBuffer) {
|
|
||||||
return unpack(data) as ArrayBuffer;
|
|
||||||
}
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Interceptor for request failure. This interceptor will be entered when the request is wrong.
|
|
||||||
// http errors like 401 (unauthorized) are handled either in the methods or AuthenticatedRouting()
|
|
||||||
// onError: (error, method) => {
|
|
||||||
// alert(error.message);
|
|
||||||
// }
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -2,12 +2,14 @@ import type { NetworkSettingsType, NetworkStatusType, WiFiNetworkList } from 'ty
|
|||||||
|
|
||||||
import { alovaInstance } from './endpoints';
|
import { alovaInstance } from './endpoints';
|
||||||
|
|
||||||
|
const LIST_NETWORKS_TIMEOUT = 20000; // 20 seconds
|
||||||
|
|
||||||
export const readNetworkStatus = () =>
|
export const readNetworkStatus = () =>
|
||||||
alovaInstance.Get<NetworkStatusType>('/rest/networkStatus');
|
alovaInstance.Get<NetworkStatusType>('/rest/networkStatus');
|
||||||
export const scanNetworks = () => alovaInstance.Get('/rest/scanNetworks');
|
export const scanNetworks = () => alovaInstance.Get('/rest/scanNetworks');
|
||||||
export const listNetworks = () =>
|
export const listNetworks = () =>
|
||||||
alovaInstance.Get<WiFiNetworkList>('/rest/listNetworks', {
|
alovaInstance.Get<WiFiNetworkList>('/rest/listNetworks', {
|
||||||
timeout: 20000 // 20 seconds
|
timeout: LIST_NETWORKS_TIMEOUT
|
||||||
});
|
});
|
||||||
export const readNetworkSettings = () =>
|
export const readNetworkSettings = () =>
|
||||||
alovaInstance.Get<NetworkSettingsType>('/rest/networkSettings');
|
alovaInstance.Get<NetworkSettingsType>('/rest/networkSettings');
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ export const readNTPStatus = () =>
|
|||||||
alovaInstance.Get<NTPStatusType>('/rest/ntpStatus');
|
alovaInstance.Get<NTPStatusType>('/rest/ntpStatus');
|
||||||
|
|
||||||
export const readNTPSettings = () =>
|
export const readNTPSettings = () =>
|
||||||
alovaInstance.Get<NTPSettingsType>('/rest/ntpSettings', {});
|
alovaInstance.Get<NTPSettingsType>('/rest/ntpSettings');
|
||||||
export const updateNTPSettings = (data: NTPSettingsType) =>
|
export const updateNTPSettings = (data: NTPSettingsType) =>
|
||||||
alovaInstance.Post<NTPSettingsType>('/rest/ntpSettings', data);
|
alovaInstance.Post<NTPSettingsType>('/rest/ntpSettings', data);
|
||||||
|
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ export const readSystemStatus = () =>
|
|||||||
|
|
||||||
// SystemLog
|
// SystemLog
|
||||||
export const readLogSettings = () =>
|
export const readLogSettings = () =>
|
||||||
alovaInstance.Get<LogSettings>(`/rest/logSettings`);
|
alovaInstance.Get<LogSettings>('/rest/logSettings');
|
||||||
export const updateLogSettings = (data: LogSettings) =>
|
export const updateLogSettings = (data: LogSettings) =>
|
||||||
alovaInstance.Post('/rest/logSettings', data);
|
alovaInstance.Post('/rest/logSettings', data);
|
||||||
export const fetchLogES = () => alovaInstance.Get('/es/log');
|
export const fetchLogES = () => alovaInstance.Get('/es/log');
|
||||||
@@ -36,10 +36,12 @@ export const getDevVersion = () =>
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const UPLOAD_TIMEOUT = 60000; // 1 minute
|
||||||
|
|
||||||
export const uploadFile = (file: File) => {
|
export const uploadFile = (file: File) => {
|
||||||
const formData = new FormData();
|
const formData = new FormData();
|
||||||
formData.append('file', file);
|
formData.append('file', file);
|
||||||
return alovaInstance.Post('/rest/uploadFile', formData, {
|
return alovaInstance.Post('/rest/uploadFile', formData, {
|
||||||
timeout: 60000 // override timeout for uploading firmware - 1 minute
|
timeout: UPLOAD_TIMEOUT
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ export class Unpackr {
|
|||||||
}
|
}
|
||||||
Object.assign(this, options);
|
Object.assign(this, options);
|
||||||
}
|
}
|
||||||
unpack(source, options?: any) {
|
unpack(source, options?: { start?: number; end?: number; lazy?: boolean }) {
|
||||||
if (src) {
|
if (src) {
|
||||||
return saveState(() => {
|
return saveState(() => {
|
||||||
clearSource();
|
clearSource();
|
||||||
@@ -184,7 +184,7 @@ export class Unpackr {
|
|||||||
function getPosition() {
|
function getPosition() {
|
||||||
return position;
|
return position;
|
||||||
}
|
}
|
||||||
function checkedRead(options: any) {
|
function checkedRead(options?: { lazy?: boolean }) {
|
||||||
try {
|
try {
|
||||||
if (!currentUnpackr.trusted && !sequentialMode) {
|
if (!currentUnpackr.trusted && !sequentialMode) {
|
||||||
const sharedLength = currentStructures.sharedLength || 0;
|
const sharedLength = currentStructures.sharedLength || 0;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { useBlocker } from 'react-router';
|
import { useBlocker } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
@@ -35,6 +35,10 @@ import { DeviceValueTypeNames, DeviceValueUOM_s } from './types';
|
|||||||
import type { Entities, EntityItem } from './types';
|
import type { Entities, EntityItem } from './types';
|
||||||
import { entityItemValidation } from './validators';
|
import { entityItemValidation } from './validators';
|
||||||
|
|
||||||
|
const MIN_ID = -100;
|
||||||
|
const MAX_ID = 100;
|
||||||
|
const ICON_SIZE = 12;
|
||||||
|
|
||||||
const CustomEntities = () => {
|
const CustomEntities = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [numChanges, setNumChanges] = useState<number>(0);
|
const [numChanges, setNumChanges] = useState<number>(0);
|
||||||
@@ -53,18 +57,20 @@ const CustomEntities = () => {
|
|||||||
initialData: []
|
initialData: []
|
||||||
});
|
});
|
||||||
|
|
||||||
useInterval(() => {
|
const intervalCallback = useCallback(() => {
|
||||||
if (!dialogOpen && !numChanges) {
|
if (!dialogOpen && !numChanges) {
|
||||||
void fetchEntities();
|
void fetchEntities();
|
||||||
}
|
}
|
||||||
});
|
}, [dialogOpen, numChanges, fetchEntities]);
|
||||||
|
|
||||||
|
useInterval(intervalCallback);
|
||||||
|
|
||||||
const { send: writeEntities } = useRequest(
|
const { send: writeEntities } = useRequest(
|
||||||
(data: Entities) => writeCustomEntities(data),
|
(data: Entities) => writeCustomEntities(data),
|
||||||
{ immediate: false }
|
{ immediate: false }
|
||||||
);
|
);
|
||||||
|
|
||||||
function hasEntityChanged(ei: EntityItem) {
|
const hasEntityChanged = useCallback((ei: EntityItem) => {
|
||||||
return (
|
return (
|
||||||
ei.id !== ei.o_id ||
|
ei.id !== ei.o_id ||
|
||||||
ei.ram !== ei.o_ram ||
|
ei.ram !== ei.o_ram ||
|
||||||
@@ -80,9 +86,11 @@ const CustomEntities = () => {
|
|||||||
ei.deleted !== ei.o_deleted ||
|
ei.deleted !== ei.o_deleted ||
|
||||||
(ei.value || '') !== (ei.o_value || '')
|
(ei.value || '') !== (ei.o_value || '')
|
||||||
);
|
);
|
||||||
}
|
}, []);
|
||||||
|
|
||||||
const entity_theme = useTheme({
|
const entity_theme = useMemo(
|
||||||
|
() =>
|
||||||
|
useTheme({
|
||||||
Table: `
|
Table: `
|
||||||
--data-table-library_grid-template-columns: repeat(1, minmax(60px, 1fr)) minmax(80px, auto) 80px 80px 80px 120px;
|
--data-table-library_grid-template-columns: repeat(1, minmax(60px, 1fr)) minmax(80px, auto) 80px 80px 80px 120px;
|
||||||
`,
|
`,
|
||||||
@@ -132,9 +140,11 @@ const CustomEntities = () => {
|
|||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
});
|
}),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
const saveEntities = async () => {
|
const saveEntities = useCallback(async () => {
|
||||||
await writeEntities({
|
await writeEntities({
|
||||||
entities: entities
|
entities: entities
|
||||||
.filter((ei: EntityItem) => !ei.deleted)
|
.filter((ei: EntityItem) => !ei.deleted)
|
||||||
@@ -163,7 +173,7 @@ const CustomEntities = () => {
|
|||||||
await fetchEntities();
|
await fetchEntities();
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
};
|
}, [entities, writeEntities, LL, fetchEntities]);
|
||||||
|
|
||||||
const editEntityItem = useCallback((ei: EntityItem) => {
|
const editEntityItem = useCallback((ei: EntityItem) => {
|
||||||
setCreating(false);
|
setCreating(false);
|
||||||
@@ -171,17 +181,18 @@ const CustomEntities = () => {
|
|||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onDialogClose = () => {
|
const onDialogClose = useCallback(() => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const onDialogCancel = async () => {
|
const onDialogCancel = useCallback(async () => {
|
||||||
await fetchEntities().then(() => {
|
await fetchEntities().then(() => {
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
};
|
}, [fetchEntities]);
|
||||||
|
|
||||||
const onDialogSave = (updatedItem: EntityItem) => {
|
const onDialogSave = useCallback(
|
||||||
|
(updatedItem: EntityItem) => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
void updateState(readCustomEntities(), (data: EntityItem[]) => {
|
void updateState(readCustomEntities(), (data: EntityItem[]) => {
|
||||||
const new_data = creating
|
const new_data = creating
|
||||||
@@ -195,12 +206,14 @@ const CustomEntities = () => {
|
|||||||
setNumChanges(new_data.filter((ei) => hasEntityChanged(ei)).length);
|
setNumChanges(new_data.filter((ei) => hasEntityChanged(ei)).length);
|
||||||
return new_data;
|
return new_data;
|
||||||
});
|
});
|
||||||
};
|
},
|
||||||
|
[creating, hasEntityChanged]
|
||||||
|
);
|
||||||
|
|
||||||
const onDialogDup = (item: EntityItem) => {
|
const onDialogDup = useCallback((item: EntityItem) => {
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
setSelectedEntityItem({
|
setSelectedEntityItem({
|
||||||
id: Math.floor(Math.random() * (Math.floor(200) - 100) + 100),
|
id: Math.floor(Math.random() * (MAX_ID - MIN_ID) + MIN_ID),
|
||||||
name: item.name + '_',
|
name: item.name + '_',
|
||||||
ram: item.ram,
|
ram: item.ram,
|
||||||
device_id: item.device_id,
|
device_id: item.device_id,
|
||||||
@@ -215,12 +228,12 @@ const CustomEntities = () => {
|
|||||||
value: item.value
|
value: item.value
|
||||||
});
|
});
|
||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const addEntityItem = () => {
|
const addEntityItem = useCallback(() => {
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
setSelectedEntityItem({
|
setSelectedEntityItem({
|
||||||
id: Math.floor(Math.random() * (Math.floor(200) - 100) + 100),
|
id: Math.floor(Math.random() * (MAX_ID - MIN_ID) + MIN_ID),
|
||||||
name: '',
|
name: '',
|
||||||
ram: 0,
|
ram: 0,
|
||||||
device_id: '0',
|
device_id: '0',
|
||||||
@@ -235,22 +248,30 @@ const CustomEntities = () => {
|
|||||||
value: ''
|
value: ''
|
||||||
});
|
});
|
||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
function formatValue(value: unknown, uom: number) {
|
const formatValue = useCallback((value: unknown, uom: number) => {
|
||||||
return value === undefined
|
return value === undefined
|
||||||
? ''
|
? ''
|
||||||
: typeof value === 'number'
|
: typeof value === 'number'
|
||||||
? new Intl.NumberFormat().format(value) +
|
? new Intl.NumberFormat().format(value) +
|
||||||
(uom === 0 ? '' : ' ' + DeviceValueUOM_s[uom])
|
(uom === 0 ? '' : ` ${DeviceValueUOM_s[uom]}`)
|
||||||
: (value as string) + (uom === 0 ? '' : ' ' + DeviceValueUOM_s[uom]);
|
: `${value as string}${uom === 0 ? '' : ` ${DeviceValueUOM_s[uom]}`}`;
|
||||||
}
|
}, []);
|
||||||
|
|
||||||
function showHex(value: number, digit: number) {
|
const showHex = useCallback((value: number, digit: number) => {
|
||||||
return '0x' + value.toString(16).toUpperCase().padStart(digit, '0');
|
return `0x${value.toString(16).toUpperCase().padStart(digit, '0')}`;
|
||||||
}
|
}, []);
|
||||||
|
|
||||||
const renderEntity = () => {
|
const filteredAndSortedEntities = useMemo(
|
||||||
|
() =>
|
||||||
|
entities
|
||||||
|
?.filter((ei: EntityItem) => !ei.deleted)
|
||||||
|
.sort((a: EntityItem, b: EntityItem) => a.name.localeCompare(b.name)) ?? [],
|
||||||
|
[entities]
|
||||||
|
);
|
||||||
|
|
||||||
|
const renderEntity = useCallback(() => {
|
||||||
if (!entities) {
|
if (!entities) {
|
||||||
return (
|
return (
|
||||||
<FormLoader onRetry={fetchEntities} errorMessage={error?.message || ''} />
|
<FormLoader onRetry={fetchEntities} errorMessage={error?.message || ''} />
|
||||||
@@ -260,9 +281,7 @@ const CustomEntities = () => {
|
|||||||
return (
|
return (
|
||||||
<Table
|
<Table
|
||||||
data={{
|
data={{
|
||||||
nodes: entities
|
nodes: filteredAndSortedEntities
|
||||||
.filter((ei: EntityItem) => !ei.deleted)
|
|
||||||
.sort((a: EntityItem, b: EntityItem) => a.name.localeCompare(b.name))
|
|
||||||
}}
|
}}
|
||||||
theme={entity_theme}
|
theme={entity_theme}
|
||||||
layout={{ custom: true }}
|
layout={{ custom: true }}
|
||||||
@@ -285,16 +304,21 @@ const CustomEntities = () => {
|
|||||||
<Cell>
|
<Cell>
|
||||||
{ei.name}
|
{ei.name}
|
||||||
{ei.writeable && (
|
{ei.writeable && (
|
||||||
<EditOutlinedIcon color="primary" sx={{ fontSize: 12 }} />
|
<EditOutlinedIcon
|
||||||
|
color="primary"
|
||||||
|
sx={{ fontSize: ICON_SIZE }}
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
</Cell>
|
</Cell>
|
||||||
|
<Cell>{ei.ram > 0 ? '' : showHex(ei.device_id as number, 2)}</Cell>
|
||||||
|
<Cell>{ei.ram > 0 ? '' : showHex(ei.type_id as number, 3)}</Cell>
|
||||||
|
<Cell>{ei.ram > 0 ? '' : ei.offset}</Cell>
|
||||||
<Cell>
|
<Cell>
|
||||||
{ei.ram === 1 ? '' : showHex(ei.device_id as number, 2)}
|
{ei.ram === 1
|
||||||
</Cell>
|
? 'RAM'
|
||||||
<Cell>{ei.ram === 1 ? '' : showHex(ei.type_id as number, 3)}</Cell>
|
: ei.ram === 2
|
||||||
<Cell>{ei.ram === 1 ? '' : ei.offset}</Cell>
|
? 'NVS'
|
||||||
<Cell>
|
: DeviceValueTypeNames[ei.value_type]}
|
||||||
{ei.ram === 1 ? 'RAM' : DeviceValueTypeNames[ei.value_type]}
|
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell>{formatValue(ei.value, ei.uom)}</Cell>
|
<Cell>{formatValue(ei.value, ei.uom)}</Cell>
|
||||||
</Row>
|
</Row>
|
||||||
@@ -304,7 +328,17 @@ const CustomEntities = () => {
|
|||||||
)}
|
)}
|
||||||
</Table>
|
</Table>
|
||||||
);
|
);
|
||||||
};
|
}, [
|
||||||
|
entities,
|
||||||
|
error,
|
||||||
|
fetchEntities,
|
||||||
|
entity_theme,
|
||||||
|
editEntityItem,
|
||||||
|
LL,
|
||||||
|
filteredAndSortedEntities,
|
||||||
|
showHex,
|
||||||
|
formatValue
|
||||||
|
]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
@@ -327,7 +361,7 @@ const CustomEntities = () => {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Box mt={1} display="flex" flexWrap="wrap">
|
<Box mt={2} display="flex" flexWrap="wrap">
|
||||||
<Box flexGrow={1}>
|
<Box flexGrow={1}>
|
||||||
{numChanges > 0 && (
|
{numChanges > 0 && (
|
||||||
<ButtonRow>
|
<ButtonRow>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import AddIcon from '@mui/icons-material/Add';
|
import AddIcon from '@mui/icons-material/Add';
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -33,6 +33,19 @@ import { validate } from 'validators';
|
|||||||
import { DeviceValueType, DeviceValueTypeNames, DeviceValueUOM_s } from './types';
|
import { DeviceValueType, DeviceValueTypeNames, DeviceValueUOM_s } from './types';
|
||||||
import type { EntityItem } from './types';
|
import type { EntityItem } from './types';
|
||||||
|
|
||||||
|
// Constant value type options for the dropdown
|
||||||
|
const VALUE_TYPE_OPTIONS = [
|
||||||
|
DeviceValueType.BOOL,
|
||||||
|
DeviceValueType.INT8,
|
||||||
|
DeviceValueType.UINT8,
|
||||||
|
DeviceValueType.INT16,
|
||||||
|
DeviceValueType.UINT16,
|
||||||
|
DeviceValueType.UINT24,
|
||||||
|
DeviceValueType.TIME,
|
||||||
|
DeviceValueType.UINT32,
|
||||||
|
DeviceValueType.STRING
|
||||||
|
] as const;
|
||||||
|
|
||||||
interface CustomEntitiesDialogProps {
|
interface CustomEntitiesDialogProps {
|
||||||
open: boolean;
|
open: boolean;
|
||||||
creating: boolean;
|
creating: boolean;
|
||||||
@@ -55,64 +68,97 @@ const CustomEntitiesDialog = ({
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [editItem, setEditItem] = useState<EntityItem>(selectedItem);
|
const [editItem, setEditItem] = useState<EntityItem>(selectedItem);
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
const updateFormValue = updateValue(setEditItem);
|
const updateFormValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValue(
|
||||||
|
setEditItem as unknown as React.Dispatch<
|
||||||
|
React.SetStateAction<Record<string, unknown>>
|
||||||
|
>
|
||||||
|
),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
setEditItem(selectedItem);
|
// Convert to hex strings - combined into single setEditItem call
|
||||||
// convert to hex strings straight away
|
const deviceIdHex =
|
||||||
|
typeof selectedItem.device_id === 'number'
|
||||||
|
? selectedItem.device_id.toString(16).toUpperCase()
|
||||||
|
: selectedItem.device_id;
|
||||||
|
const typeIdHex =
|
||||||
|
typeof selectedItem.type_id === 'number'
|
||||||
|
? selectedItem.type_id.toString(16).toUpperCase()
|
||||||
|
: selectedItem.type_id;
|
||||||
|
const factorValue =
|
||||||
|
selectedItem.value_type === DeviceValueType.BOOL &&
|
||||||
|
typeof selectedItem.factor === 'number'
|
||||||
|
? selectedItem.factor.toString(16).toUpperCase()
|
||||||
|
: selectedItem.factor;
|
||||||
|
|
||||||
setEditItem({
|
setEditItem({
|
||||||
...selectedItem,
|
...selectedItem,
|
||||||
device_id: selectedItem.device_id.toString(16).toUpperCase(),
|
device_id: deviceIdHex,
|
||||||
type_id: selectedItem.type_id.toString(16).toUpperCase(),
|
type_id: typeIdHex,
|
||||||
factor:
|
factor: factorValue
|
||||||
selectedItem.value_type === DeviceValueType.BOOL
|
|
||||||
? selectedItem.factor.toString(16).toUpperCase()
|
|
||||||
: selectedItem.factor
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleClose = (
|
const handleClose = useCallback(
|
||||||
_event: React.SyntheticEvent,
|
(_event: React.SyntheticEvent, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
reason: 'backdropClick' | 'escapeKeyDown'
|
|
||||||
) => {
|
|
||||||
if (reason !== 'backdropClick') {
|
if (reason !== 'backdropClick') {
|
||||||
onClose();
|
onClose();
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[onClose]
|
||||||
|
);
|
||||||
|
|
||||||
const save = async () => {
|
const save = useCallback(async () => {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, editItem);
|
||||||
if (typeof editItem.device_id === 'string') {
|
|
||||||
editItem.device_id = parseInt(editItem.device_id, 16);
|
// Create a copy to avoid mutating the state directly
|
||||||
|
const processedItem: EntityItem = { ...editItem };
|
||||||
|
|
||||||
|
if (typeof processedItem.device_id === 'string') {
|
||||||
|
processedItem.device_id = Number.parseInt(processedItem.device_id, 16);
|
||||||
}
|
}
|
||||||
if (typeof editItem.type_id === 'string') {
|
if (typeof processedItem.type_id === 'string') {
|
||||||
editItem.type_id = parseInt(editItem.type_id, 16);
|
processedItem.type_id = Number.parseInt(processedItem.type_id, 16);
|
||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
editItem.value_type === DeviceValueType.BOOL &&
|
processedItem.value_type === DeviceValueType.BOOL &&
|
||||||
typeof editItem.factor === 'string'
|
typeof processedItem.factor === 'string'
|
||||||
) {
|
) {
|
||||||
editItem.factor = parseInt(editItem.factor, 16);
|
processedItem.factor = Number.parseInt(processedItem.factor, 16);
|
||||||
}
|
}
|
||||||
onSave(editItem);
|
onSave(processedItem);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
}, [validator, editItem, onSave]);
|
||||||
|
|
||||||
const remove = () => {
|
const remove = useCallback(() => {
|
||||||
editItem.deleted = true;
|
const itemWithDeleted = { ...editItem, deleted: true };
|
||||||
onSave(editItem);
|
onSave(itemWithDeleted);
|
||||||
};
|
}, [editItem, onSave]);
|
||||||
|
|
||||||
const dup = () => {
|
const dup = useCallback(() => {
|
||||||
onDup(editItem);
|
onDup(editItem);
|
||||||
};
|
}, [editItem, onDup]);
|
||||||
|
|
||||||
|
// Memoize UOM menu items to avoid recreating on every render
|
||||||
|
const uomMenuItems = useMemo(
|
||||||
|
() =>
|
||||||
|
DeviceValueUOM_s.map((val, i) => (
|
||||||
|
<MenuItem key={val} value={i}>
|
||||||
|
{val}
|
||||||
|
</MenuItem>
|
||||||
|
)),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
@@ -120,9 +166,6 @@ const CustomEntitiesDialog = ({
|
|||||||
{creating ? LL.ADD(1) + ' ' + LL.NEW(1) : LL.EDIT()} {LL.ENTITY()}
|
{creating ? LL.ADD(1) + ' ' + LL.NEW(1) : LL.EDIT()} {LL.ENTITY()}
|
||||||
</DialogTitle>
|
</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Box display="flex" flexWrap="wrap" mb={1}>
|
|
||||||
<Box flexWrap="nowrap" whiteSpace="nowrap" />
|
|
||||||
</Box>
|
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid size={12}>
|
<Grid size={12}>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
@@ -162,9 +205,10 @@ const CustomEntitiesDialog = ({
|
|||||||
>
|
>
|
||||||
<MenuItem value={0}>EMS-{LL.VALUE(1)}</MenuItem>
|
<MenuItem value={0}>EMS-{LL.VALUE(1)}</MenuItem>
|
||||||
<MenuItem value={1}>RAM-{LL.VALUE(1)}</MenuItem>
|
<MenuItem value={1}>RAM-{LL.VALUE(1)}</MenuItem>
|
||||||
|
<MenuItem value={2}>NVS-{LL.VALUE(1)}</MenuItem>
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
{editItem.ram === 1 && (
|
{editItem.ram > 0 && (
|
||||||
<>
|
<>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<TextField
|
||||||
@@ -187,11 +231,7 @@ const CustomEntitiesDialog = ({
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
{DeviceValueUOM_s.map((val, i) => (
|
{uomMenuItems}
|
||||||
<MenuItem key={val} value={i}>
|
|
||||||
{val}
|
|
||||||
</MenuItem>
|
|
||||||
))}
|
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
</>
|
</>
|
||||||
@@ -275,33 +315,11 @@ const CustomEntitiesDialog = ({
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
<MenuItem value={DeviceValueType.BOOL}>
|
{VALUE_TYPE_OPTIONS.map((valueType) => (
|
||||||
{DeviceValueTypeNames[DeviceValueType.BOOL]}
|
<MenuItem key={valueType} value={valueType}>
|
||||||
</MenuItem>
|
{DeviceValueTypeNames[valueType]}
|
||||||
<MenuItem value={DeviceValueType.INT8}>
|
|
||||||
{DeviceValueTypeNames[DeviceValueType.INT8]}
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem value={DeviceValueType.UINT8}>
|
|
||||||
{DeviceValueTypeNames[DeviceValueType.UINT8]}
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem value={DeviceValueType.INT16}>
|
|
||||||
{DeviceValueTypeNames[DeviceValueType.INT16]}
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem value={DeviceValueType.UINT16}>
|
|
||||||
{DeviceValueTypeNames[DeviceValueType.UINT16]}
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem value={DeviceValueType.UINT24}>
|
|
||||||
{DeviceValueTypeNames[DeviceValueType.UINT24]}
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem value={DeviceValueType.TIME}>
|
|
||||||
{DeviceValueTypeNames[DeviceValueType.TIME]}
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem value={DeviceValueType.UINT32}>
|
|
||||||
{DeviceValueTypeNames[DeviceValueType.UINT32]}
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem value={DeviceValueType.STRING}>
|
|
||||||
{DeviceValueTypeNames[DeviceValueType.STRING]}
|
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
|
))}
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
@@ -333,11 +351,7 @@ const CustomEntitiesDialog = ({
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
{DeviceValueUOM_s.map((val, i) => (
|
{uomMenuItems}
|
||||||
<MenuItem key={val} value={i}>
|
|
||||||
{val}
|
|
||||||
</MenuItem>
|
|
||||||
))}
|
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
import { useBlocker, useLocation } from 'react-router';
|
import { useBlocker, useLocation } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
@@ -62,7 +62,24 @@ import OptionIcon from './OptionIcon';
|
|||||||
import { DeviceEntityMask } from './types';
|
import { DeviceEntityMask } from './types';
|
||||||
import type { APIcall, Device, DeviceEntity } from './types';
|
import type { APIcall, Device, DeviceEntity } from './types';
|
||||||
|
|
||||||
export const APIURL = window.location.origin + '/api/';
|
export const APIURL = `${window.location.origin}/api/`;
|
||||||
|
|
||||||
|
const MAX_BUFFER_SIZE = 2000;
|
||||||
|
|
||||||
|
// Helper function to create masked entity ID - extracted to avoid duplication
|
||||||
|
const createMaskedEntityId = (de: DeviceEntity): string => {
|
||||||
|
const maskHex = de.m.toString(16).padStart(2, '0');
|
||||||
|
const hasCustomizations = !!(de.cn || de.mi || de.ma);
|
||||||
|
const customizations = [
|
||||||
|
de.cn || '',
|
||||||
|
de.mi ? `>${de.mi}` : '',
|
||||||
|
de.ma ? `<${de.ma}` : ''
|
||||||
|
]
|
||||||
|
.filter(Boolean)
|
||||||
|
.join('');
|
||||||
|
|
||||||
|
return `${maskHex}${de.id}${hasCustomizations ? `|${customizations}` : ''}`;
|
||||||
|
};
|
||||||
|
|
||||||
const Customizations = () => {
|
const Customizations = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
@@ -94,13 +111,14 @@ const Customizations = () => {
|
|||||||
const [selectedDeviceTypeNameURL, setSelectedDeviceTypeNameURL] =
|
const [selectedDeviceTypeNameURL, setSelectedDeviceTypeNameURL] =
|
||||||
useState<string>(''); // needed for API URL
|
useState<string>(''); // needed for API URL
|
||||||
const [selectedDeviceName, setSelectedDeviceName] = useState<string>('');
|
const [selectedDeviceName, setSelectedDeviceName] = useState<string>('');
|
||||||
|
const [selectedDeviceBrand, setSelectedDeviceBrand] = useState<string>('');
|
||||||
|
|
||||||
const { send: sendResetCustomizations } = useRequest(resetCustomizations(), {
|
const { send: sendResetCustomizations } = useRequest(resetCustomizations(), {
|
||||||
immediate: false
|
immediate: false
|
||||||
});
|
});
|
||||||
|
|
||||||
const { send: sendDeviceName } = useRequest(
|
const { send: sendDeviceName } = useRequest(
|
||||||
(data: { id: number; name: string }) => writeDeviceName(data),
|
(data: { id: number; name: string; brand: string }) => writeDeviceName(data),
|
||||||
{
|
{
|
||||||
immediate: false
|
immediate: false
|
||||||
}
|
}
|
||||||
@@ -153,7 +171,9 @@ const Customizations = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const entities_theme = useTheme({
|
const entities_theme = useMemo(
|
||||||
|
() =>
|
||||||
|
useTheme({
|
||||||
Table: `
|
Table: `
|
||||||
--data-table-library_grid-template-columns: 156px repeat(1, minmax(80px, 1fr)) 45px minmax(45px, auto) minmax(120px, auto);
|
--data-table-library_grid-template-columns: 156px repeat(1, minmax(80px, 1fr)) 45px minmax(45px, auto) minmax(120px, auto);
|
||||||
`,
|
`,
|
||||||
@@ -216,7 +236,9 @@ const Customizations = () => {
|
|||||||
padding-right: 8px;
|
padding-right: 8px;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
});
|
}),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
function hasEntityChanged(de: DeviceEntity) {
|
function hasEntityChanged(de: DeviceEntity) {
|
||||||
return (
|
return (
|
||||||
@@ -229,19 +251,8 @@ const Customizations = () => {
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (deviceEntities.length) {
|
if (deviceEntities.length) {
|
||||||
setNumChanges(
|
const changedEntities = deviceEntities.filter((de) => hasEntityChanged(de));
|
||||||
deviceEntities
|
setNumChanges(changedEntities.length);
|
||||||
.filter((de) => hasEntityChanged(de))
|
|
||||||
.map(
|
|
||||||
(new_de) =>
|
|
||||||
new_de.m.toString(16).padStart(2, '0') +
|
|
||||||
new_de.id +
|
|
||||||
(new_de.cn || new_de.mi || new_de.ma ? '|' : '') +
|
|
||||||
(new_de.cn ? new_de.cn : '') +
|
|
||||||
(new_de.mi ? '>' + new_de.mi : '') +
|
|
||||||
(new_de.ma ? '<' + new_de.ma : '')
|
|
||||||
).length
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}, [deviceEntities]);
|
}, [deviceEntities]);
|
||||||
|
|
||||||
@@ -257,6 +268,7 @@ const Customizations = () => {
|
|||||||
if (device) {
|
if (device) {
|
||||||
setSelectedDeviceTypeNameURL(device.url || '');
|
setSelectedDeviceTypeNameURL(device.url || '');
|
||||||
setSelectedDeviceName(device.n);
|
setSelectedDeviceName(device.n);
|
||||||
|
setSelectedDeviceBrand(device.b);
|
||||||
}
|
}
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
setRestartNeeded(false);
|
setRestartNeeded(false);
|
||||||
@@ -275,18 +287,26 @@ const Customizations = () => {
|
|||||||
return value as string;
|
return value as string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const formatName = (de: DeviceEntity, withShortname: boolean) =>
|
const isCommand = useCallback((de: DeviceEntity) => {
|
||||||
(de.n && de.n[0] === '!'
|
return de.n && de.n[0] === '!';
|
||||||
? de.t
|
}, []);
|
||||||
? LL.COMMAND(1) + ': ' + de.t + ' ' + de.n.slice(1)
|
|
||||||
: LL.COMMAND(1) + ': ' + de.n.slice(1)
|
const formatName = useCallback(
|
||||||
: de.cn && de.cn !== ''
|
(de: DeviceEntity, withShortname: boolean) => {
|
||||||
? de.t
|
let name: string;
|
||||||
? de.t + ' ' + de.cn
|
if (isCommand(de)) {
|
||||||
: de.cn
|
name = de.t
|
||||||
: de.t
|
? `${LL.COMMAND(1)}: ${de.t} ${de.n?.slice(1)}`
|
||||||
? de.t + ' ' + de.n
|
: `${LL.COMMAND(1)}: ${de.n?.slice(1)}`;
|
||||||
: de.n) + (withShortname ? ' ' + de.id : '');
|
} else if (de.cn && de.cn !== '') {
|
||||||
|
name = de.t ? `${de.t} ${de.cn}` : de.cn;
|
||||||
|
} else {
|
||||||
|
name = de.t ? `${de.t} ${de.n}` : de.n || '';
|
||||||
|
}
|
||||||
|
return withShortname ? `${name} ${de.id}` : name;
|
||||||
|
},
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
const getMaskNumber = (newMask: string[]) => {
|
const getMaskNumber = (newMask: string[]) => {
|
||||||
let new_mask = 0;
|
let new_mask = 0;
|
||||||
@@ -316,34 +336,33 @@ const Customizations = () => {
|
|||||||
return new_masks;
|
return new_masks;
|
||||||
};
|
};
|
||||||
|
|
||||||
const filter_entity = (de: DeviceEntity) =>
|
const filter_entity = useCallback(
|
||||||
|
(de: DeviceEntity) =>
|
||||||
(de.m & selectedFilters || !selectedFilters) &&
|
(de.m & selectedFilters || !selectedFilters) &&
|
||||||
formatName(de, true).toLowerCase().includes(search.toLowerCase());
|
formatName(de, true).toLowerCase().includes(search.toLowerCase()),
|
||||||
|
[selectedFilters, search, formatName]
|
||||||
|
);
|
||||||
|
|
||||||
const maskDisabled = (set: boolean) => {
|
const maskDisabled = useCallback(
|
||||||
setDeviceEntities(
|
(set: boolean) => {
|
||||||
deviceEntities.map(function (de) {
|
setDeviceEntities((prev) =>
|
||||||
|
prev.map((de) => {
|
||||||
if (filter_entity(de)) {
|
if (filter_entity(de)) {
|
||||||
|
const excludeMask =
|
||||||
|
DeviceEntityMask.DV_API_MQTT_EXCLUDE | DeviceEntityMask.DV_WEB_EXCLUDE;
|
||||||
return {
|
return {
|
||||||
...de,
|
...de,
|
||||||
m: set
|
m: set ? de.m | excludeMask : de.m & ~excludeMask
|
||||||
? de.m |
|
|
||||||
(DeviceEntityMask.DV_API_MQTT_EXCLUDE |
|
|
||||||
DeviceEntityMask.DV_WEB_EXCLUDE)
|
|
||||||
: de.m &
|
|
||||||
~(
|
|
||||||
DeviceEntityMask.DV_API_MQTT_EXCLUDE |
|
|
||||||
DeviceEntityMask.DV_WEB_EXCLUDE
|
|
||||||
)
|
|
||||||
};
|
};
|
||||||
} else {
|
|
||||||
return de;
|
|
||||||
}
|
}
|
||||||
|
return de;
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
};
|
},
|
||||||
|
[filter_entity]
|
||||||
|
);
|
||||||
|
|
||||||
const resetCustomization = async () => {
|
const resetCustomization = useCallback(async () => {
|
||||||
try {
|
try {
|
||||||
await sendResetCustomizations();
|
await sendResetCustomizations();
|
||||||
toast.info(LL.CUSTOMIZATIONS_RESTART());
|
toast.info(LL.CUSTOMIZATIONS_RESTART());
|
||||||
@@ -351,25 +370,30 @@ const Customizations = () => {
|
|||||||
toast.error((error as Error).message);
|
toast.error((error as Error).message);
|
||||||
} finally {
|
} finally {
|
||||||
setConfirmReset(false);
|
setConfirmReset(false);
|
||||||
|
setRestarting(true);
|
||||||
}
|
}
|
||||||
};
|
}, [sendResetCustomizations, LL]);
|
||||||
|
|
||||||
const onDialogClose = () => {
|
const onDialogClose = () => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
const updateDeviceEntity = (updatedItem: DeviceEntity) => {
|
const updateDeviceEntity = useCallback((updatedItem: DeviceEntity) => {
|
||||||
setDeviceEntities(
|
setDeviceEntities(
|
||||||
deviceEntities?.map((de) =>
|
(prev) =>
|
||||||
|
prev?.map((de) =>
|
||||||
de.id === updatedItem.id ? { ...de, ...updatedItem } : de
|
de.id === updatedItem.id ? { ...de, ...updatedItem } : de
|
||||||
)
|
) ?? []
|
||||||
);
|
);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const onDialogSave = (updatedItem: DeviceEntity) => {
|
const onDialogSave = useCallback(
|
||||||
|
(updatedItem: DeviceEntity) => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
updateDeviceEntity(updatedItem);
|
updateDeviceEntity(updatedItem);
|
||||||
};
|
},
|
||||||
|
[updateDeviceEntity]
|
||||||
|
);
|
||||||
|
|
||||||
const editDeviceEntity = useCallback((de: DeviceEntity) => {
|
const editDeviceEntity = useCallback((de: DeviceEntity) => {
|
||||||
if (de.n === undefined || (de.n && de.n[0] === '!')) {
|
if (de.n === undefined || (de.n && de.n[0] === '!')) {
|
||||||
@@ -384,23 +408,18 @@ const Customizations = () => {
|
|||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const saveCustomization = async () => {
|
const saveCustomization = useCallback(async () => {
|
||||||
if (devices && deviceEntities && selectedDevice !== -1) {
|
if (!devices || !deviceEntities || selectedDevice === -1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const masked_entities = deviceEntities
|
const masked_entities = deviceEntities
|
||||||
.filter((de: DeviceEntity) => hasEntityChanged(de))
|
.filter((de: DeviceEntity) => hasEntityChanged(de))
|
||||||
.map(
|
.map((new_de) => createMaskedEntityId(new_de));
|
||||||
(new_de) =>
|
|
||||||
new_de.m.toString(16).padStart(2, '0') +
|
|
||||||
new_de.id +
|
|
||||||
(new_de.cn || new_de.mi || new_de.ma ? '|' : '') +
|
|
||||||
(new_de.cn ? new_de.cn : '') +
|
|
||||||
(new_de.mi ? '>' + new_de.mi : '') +
|
|
||||||
(new_de.ma ? '<' + new_de.ma : '')
|
|
||||||
);
|
|
||||||
|
|
||||||
// check size in bytes to match buffer in CPP, which is 2048
|
// check size in bytes to match buffer in CPP, which is 2048
|
||||||
const bytes = new TextEncoder().encode(JSON.stringify(masked_entities)).length;
|
const bytes = new TextEncoder().encode(JSON.stringify(masked_entities)).length;
|
||||||
if (bytes > 2000) {
|
if (bytes > MAX_BUFFER_SIZE) {
|
||||||
toast.warning(LL.CUSTOMIZATIONS_FULL());
|
toast.warning(LL.CUSTOMIZATIONS_FULL());
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -422,22 +441,32 @@ const Customizations = () => {
|
|||||||
.finally(() => {
|
.finally(() => {
|
||||||
setOriginalSettings(deviceEntities);
|
setOriginalSettings(deviceEntities);
|
||||||
});
|
});
|
||||||
}
|
}, [devices, deviceEntities, selectedDevice, sendCustomizationEntities, LL]);
|
||||||
};
|
|
||||||
|
|
||||||
const renameDevice = async () => {
|
const renameDevice = useCallback(async () => {
|
||||||
await sendDeviceName({ id: selectedDevice, name: selectedDeviceName })
|
await sendDeviceName({
|
||||||
|
id: selectedDevice,
|
||||||
|
name: selectedDeviceName,
|
||||||
|
brand: selectedDeviceBrand
|
||||||
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success(LL.UPDATED_OF(LL.NAME(1)));
|
toast.success(LL.UPDATED_OF(LL.NAME(1)));
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
toast.error(LL.UPDATE_OF(LL.NAME(1)) + ' ' + LL.FAILED(1));
|
toast.error(`${LL.UPDATE_OF(LL.NAME(1))} ${LL.FAILED(1)}`);
|
||||||
})
|
})
|
||||||
.finally(async () => {
|
.finally(async () => {
|
||||||
setRename(false);
|
setRename(false);
|
||||||
await fetchCoreData();
|
await fetchCoreData();
|
||||||
});
|
});
|
||||||
};
|
}, [
|
||||||
|
selectedDevice,
|
||||||
|
selectedDeviceName,
|
||||||
|
selectedDeviceBrand,
|
||||||
|
sendDeviceName,
|
||||||
|
LL,
|
||||||
|
fetchCoreData
|
||||||
|
]);
|
||||||
|
|
||||||
const renderDeviceList = () => (
|
const renderDeviceList = () => (
|
||||||
<>
|
<>
|
||||||
@@ -446,15 +475,26 @@ const Customizations = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
<Box display="flex" flexWrap="wrap" alignItems="center" gap={2}>
|
<Box display="flex" flexWrap="wrap" alignItems="center" gap={2}>
|
||||||
{rename ? (
|
{rename ? (
|
||||||
|
<>
|
||||||
<TextField
|
<TextField
|
||||||
name="device"
|
name="device"
|
||||||
label={LL.EMS_DEVICE()}
|
label={LL.EMS_DEVICE()}
|
||||||
fullWidth
|
style={{ minWidth: '48%' }}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={selectedDeviceName}
|
value={selectedDeviceName}
|
||||||
onChange={(e) => setSelectedDeviceName(e.target.value)}
|
onChange={(e) => setSelectedDeviceName(e.target.value)}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
|
<TextField
|
||||||
|
name="brand"
|
||||||
|
label={LL.BRAND()}
|
||||||
|
style={{ minWidth: '48%' }}
|
||||||
|
variant="outlined"
|
||||||
|
value={selectedDeviceBrand}
|
||||||
|
onChange={(e) => setSelectedDeviceBrand(e.target.value)}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</>
|
||||||
) : (
|
) : (
|
||||||
<TextField
|
<TextField
|
||||||
name="device"
|
name="device"
|
||||||
@@ -500,6 +540,7 @@ const Customizations = () => {
|
|||||||
</Button>
|
</Button>
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
|
<>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<EditIcon />}
|
startIcon={<EditIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -507,18 +548,30 @@ const Customizations = () => {
|
|||||||
>
|
>
|
||||||
{LL.RENAME()}
|
{LL.RENAME()}
|
||||||
</Button>
|
</Button>
|
||||||
|
<Button
|
||||||
|
startIcon={<SettingsBackupRestoreIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
color="error"
|
||||||
|
onClick={() => setConfirmReset(true)}
|
||||||
|
>
|
||||||
|
{LL.REMOVE_ALL()}
|
||||||
|
</Button>
|
||||||
|
</>
|
||||||
))}
|
))}
|
||||||
</Box>
|
</Box>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
||||||
const renderDeviceData = () => {
|
const filteredEntities = useMemo(
|
||||||
const shown_data = deviceEntities.filter((de) => filter_entity(de));
|
() => deviceEntities.filter((de) => filter_entity(de)),
|
||||||
|
[deviceEntities, filter_entity]
|
||||||
|
);
|
||||||
|
|
||||||
|
const renderDeviceData = () => {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Box color="warning.main">
|
<Box color="warning.main">
|
||||||
<Typography variant="body2" mt={1}>
|
<Typography variant="body2" mt={1} mb={1}>
|
||||||
<OptionIcon type="favorite" isSet={true} />={LL.CUSTOMIZATIONS_HELP_2()}
|
<OptionIcon type="favorite" isSet={true} />={LL.CUSTOMIZATIONS_HELP_2()}
|
||||||
|
|
||||||
<OptionIcon type="readonly" isSet={true} />={LL.CUSTOMIZATIONS_HELP_3()}
|
<OptionIcon type="readonly" isSet={true} />={LL.CUSTOMIZATIONS_HELP_3()}
|
||||||
@@ -544,6 +597,7 @@ const Customizations = () => {
|
|||||||
size="small"
|
size="small"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
placeholder={LL.SEARCH()}
|
placeholder={LL.SEARCH()}
|
||||||
|
aria-label={LL.SEARCH()}
|
||||||
onChange={(event) => {
|
onChange={(event) => {
|
||||||
setSearch(event.target.value);
|
setSearch(event.target.value);
|
||||||
}}
|
}}
|
||||||
@@ -612,13 +666,13 @@ const Customizations = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<Typography variant="subtitle2" color="grey">
|
<Typography variant="subtitle2" color="grey">
|
||||||
{LL.SHOWING()} {shown_data.length}/{deviceEntities.length}
|
{LL.SHOWING()} {filteredEntities.length}/{deviceEntities.length}
|
||||||
{LL.ENTITIES(deviceEntities.length)}
|
{LL.ENTITIES(deviceEntities.length)}
|
||||||
</Typography>
|
</Typography>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Table
|
<Table
|
||||||
data={{ nodes: shown_data }}
|
data={{ nodes: filteredEntities }}
|
||||||
theme={entities_theme}
|
theme={entities_theme}
|
||||||
layout={{ custom: true }}
|
layout={{ custom: true }}
|
||||||
>
|
>
|
||||||
@@ -640,14 +694,27 @@ const Customizations = () => {
|
|||||||
<EntityMaskToggle onUpdate={updateDeviceEntity} de={de} />
|
<EntityMaskToggle onUpdate={updateDeviceEntity} de={de} />
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell>
|
<Cell>
|
||||||
|
<span
|
||||||
|
style={{
|
||||||
|
color:
|
||||||
|
de.v === undefined && !isCommand(de) ? 'grey' : 'inherit'
|
||||||
|
}}
|
||||||
|
>
|
||||||
{formatName(de, false)} (
|
{formatName(de, false)} (
|
||||||
<Link
|
<Link
|
||||||
|
style={{
|
||||||
|
color:
|
||||||
|
de.v === undefined && !isCommand(de)
|
||||||
|
? 'grey'
|
||||||
|
: 'primary'
|
||||||
|
}}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
href={APIURL + selectedDeviceTypeNameURL + '/' + de.id}
|
href={APIURL + selectedDeviceTypeNameURL + '/' + de.id}
|
||||||
>
|
>
|
||||||
{de.id}
|
{de.id}
|
||||||
</Link>
|
</Link>
|
||||||
)
|
)
|
||||||
|
</span>
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell>
|
<Cell>
|
||||||
{!(de.m & DeviceEntityMask.DV_READONLY) && formatValue(de.mi)}
|
{!(de.m & DeviceEntityMask.DV_READONLY) && formatValue(de.mi)}
|
||||||
@@ -672,7 +739,7 @@ const Customizations = () => {
|
|||||||
open={confirmReset}
|
open={confirmReset}
|
||||||
onClose={() => setConfirmReset(false)}
|
onClose={() => setConfirmReset(false)}
|
||||||
>
|
>
|
||||||
<DialogTitle>{LL.RESET(1)}</DialogTitle>
|
<DialogTitle>{LL.REMOVE_ALL()}</DialogTitle>
|
||||||
<DialogContent dividers>{LL.CUSTOMIZATIONS_RESET()}</DialogContent>
|
<DialogContent dividers>{LL.CUSTOMIZATIONS_RESET()}</DialogContent>
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
<Button
|
<Button
|
||||||
@@ -689,7 +756,7 @@ const Customizations = () => {
|
|||||||
onClick={resetCustomization}
|
onClick={resetCustomization}
|
||||||
color="error"
|
color="error"
|
||||||
>
|
>
|
||||||
{LL.RESET(0)}
|
{LL.REMOVE_ALL()}
|
||||||
</Button>
|
</Button>
|
||||||
</DialogActions>
|
</DialogActions>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
@@ -700,8 +767,9 @@ const Customizations = () => {
|
|||||||
{devices && renderDeviceList()}
|
{devices && renderDeviceList()}
|
||||||
{selectedDevice !== -1 && !rename && renderDeviceData()}
|
{selectedDevice !== -1 && !rename && renderDeviceData()}
|
||||||
{restartNeeded ? (
|
{restartNeeded ? (
|
||||||
<MessageBox my={2} level="warning" message={LL.RESTART_TEXT(0)}>
|
<MessageBox level="warning" message={LL.RESTART_TEXT(0)}>
|
||||||
<Button
|
<Button
|
||||||
|
sx={{ ml: 2 }}
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
variant="contained"
|
variant="contained"
|
||||||
color="error"
|
color="error"
|
||||||
@@ -719,7 +787,11 @@ const Customizations = () => {
|
|||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="secondary"
|
color="secondary"
|
||||||
onClick={() => devices && sendDeviceEntities(selectedDevice)}
|
onClick={() => {
|
||||||
|
if (devices) {
|
||||||
|
void sendDeviceEntities(selectedDevice);
|
||||||
|
}
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -734,28 +806,18 @@ const Customizations = () => {
|
|||||||
</ButtonRow>
|
</ButtonRow>
|
||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
{!rename && (
|
|
||||||
<ButtonRow mt={1}>
|
|
||||||
<Button
|
|
||||||
startIcon={<SettingsBackupRestoreIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
color="error"
|
|
||||||
onClick={() => setConfirmReset(true)}
|
|
||||||
>
|
|
||||||
{LL.RESET(0)}
|
|
||||||
</Button>
|
|
||||||
</ButtonRow>
|
|
||||||
)}
|
|
||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
{renderResetDialog()}
|
{renderResetDialog()}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return restarting ? (
|
||||||
|
<SystemMonitor />
|
||||||
|
) : (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{restarting ? <SystemMonitor /> : renderContent()}
|
{renderContent()}
|
||||||
{selectedDeviceEntity && (
|
{selectedDeviceEntity && (
|
||||||
<SettingsCustomizationsDialog
|
<SettingsCustomizationsDialog
|
||||||
open={dialogOpen}
|
open={dialogOpen}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { memo, useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import CloseIcon from '@mui/icons-material/Close';
|
import CloseIcon from '@mui/icons-material/Close';
|
||||||
@@ -30,6 +30,23 @@ interface SettingsCustomizationsDialogProps {
|
|||||||
selectedItem: DeviceEntity;
|
selectedItem: DeviceEntity;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface LabelValueProps {
|
||||||
|
label: string;
|
||||||
|
value: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const LabelValue = memo(({ label, value }: LabelValueProps) => (
|
||||||
|
<Grid container direction="row">
|
||||||
|
<Typography variant="body2" color="warning.main">
|
||||||
|
{label}:
|
||||||
|
</Typography>
|
||||||
|
<Typography variant="body2">{value}</Typography>
|
||||||
|
</Grid>
|
||||||
|
));
|
||||||
|
LabelValue.displayName = 'LabelValue';
|
||||||
|
|
||||||
|
const ICON_SIZE = 16;
|
||||||
|
|
||||||
const CustomizationsDialog = ({
|
const CustomizationsDialog = ({
|
||||||
open,
|
open,
|
||||||
onClose,
|
onClose,
|
||||||
@@ -40,12 +57,23 @@ const CustomizationsDialog = ({
|
|||||||
const [editItem, setEditItem] = useState<DeviceEntity>(selectedItem);
|
const [editItem, setEditItem] = useState<DeviceEntity>(selectedItem);
|
||||||
const [error, setError] = useState<boolean>(false);
|
const [error, setError] = useState<boolean>(false);
|
||||||
|
|
||||||
const updateFormValue = updateValue(setEditItem);
|
const updateFormValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValue(
|
||||||
|
setEditItem as unknown as React.Dispatch<
|
||||||
|
React.SetStateAction<Record<string, unknown>>
|
||||||
|
>
|
||||||
|
),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
const isWriteableNumber =
|
const isWriteableNumber = useMemo(
|
||||||
|
() =>
|
||||||
typeof editItem.v === 'number' &&
|
typeof editItem.v === 'number' &&
|
||||||
editItem.w &&
|
editItem.w &&
|
||||||
!(editItem.m & DeviceEntityMask.DV_READONLY);
|
!(editItem.m & DeviceEntityMask.DV_READONLY),
|
||||||
|
[editItem.v, editItem.w, editItem.m]
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
@@ -54,66 +82,59 @@ const CustomizationsDialog = ({
|
|||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleClose = (
|
const handleClose = useCallback(
|
||||||
_event: React.SyntheticEvent,
|
(_event: React.SyntheticEvent, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
reason: 'backdropClick' | 'escapeKeyDown'
|
|
||||||
) => {
|
|
||||||
if (reason !== 'backdropClick') {
|
if (reason !== 'backdropClick') {
|
||||||
onClose();
|
onClose();
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[onClose]
|
||||||
|
);
|
||||||
|
|
||||||
const save = () => {
|
const save = useCallback(() => {
|
||||||
if (
|
if (
|
||||||
isWriteableNumber &&
|
isWriteableNumber &&
|
||||||
editItem.mi &&
|
editItem.mi &&
|
||||||
editItem.ma &&
|
editItem.ma &&
|
||||||
editItem.mi > editItem?.ma
|
editItem.mi > editItem.ma
|
||||||
) {
|
) {
|
||||||
setError(true);
|
setError(true);
|
||||||
} else {
|
} else {
|
||||||
onSave(editItem);
|
onSave(editItem);
|
||||||
}
|
}
|
||||||
};
|
}, [isWriteableNumber, editItem, onSave]);
|
||||||
|
|
||||||
const updateDeviceEntity = (updatedItem: DeviceEntity) => {
|
const updateDeviceEntity = useCallback((updatedItem: DeviceEntity) => {
|
||||||
setEditItem({ ...editItem, m: updatedItem.m });
|
setEditItem((prev) => ({ ...prev, m: updatedItem.m }));
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
|
const dialogTitle = useMemo(() => `${LL.EDIT()} ${LL.ENTITY()}`, [LL]);
|
||||||
|
|
||||||
|
const writeableIcon = useMemo(
|
||||||
|
() =>
|
||||||
|
editItem.w ? (
|
||||||
|
<DoneIcon color="success" sx={{ fontSize: ICON_SIZE }} />
|
||||||
|
) : (
|
||||||
|
<CloseIcon color="error" sx={{ fontSize: ICON_SIZE }} />
|
||||||
|
),
|
||||||
|
[editItem.w]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
<DialogTitle>{LL.EDIT() + ' ' + LL.ENTITY()}</DialogTitle>
|
<DialogTitle>{dialogTitle}</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Grid container>
|
<LabelValue label={LL.ID_OF(LL.ENTITY())} value={editItem.id} />
|
||||||
<Typography variant="body2" color="warning.main">
|
<LabelValue
|
||||||
{LL.ID_OF(LL.ENTITY())}:
|
label={`${LL.DEFAULT(1)} ${LL.ENTITY_NAME(1)}`}
|
||||||
</Typography>
|
value={editItem.n}
|
||||||
<Typography variant="body2">{editItem.id}</Typography>
|
/>
|
||||||
</Grid>
|
<LabelValue label={LL.WRITEABLE()} value={writeableIcon} />
|
||||||
|
|
||||||
<Grid container direction="row">
|
|
||||||
<Typography variant="body2" color="warning.main">
|
|
||||||
{LL.DEFAULT(1) + ' ' + LL.ENTITY_NAME(1)}:
|
|
||||||
</Typography>
|
|
||||||
<Typography variant="body2">{editItem.n}</Typography>
|
|
||||||
</Grid>
|
|
||||||
|
|
||||||
<Grid container direction="row">
|
|
||||||
<Typography variant="body2" color="warning.main">
|
|
||||||
{LL.WRITEABLE()}:
|
|
||||||
</Typography>
|
|
||||||
<Typography variant="body2">
|
|
||||||
{editItem.w ? (
|
|
||||||
<DoneIcon color="success" sx={{ fontSize: 16 }} />
|
|
||||||
) : (
|
|
||||||
<CloseIcon color="error" sx={{ fontSize: 16 }} />
|
|
||||||
)}
|
|
||||||
</Typography>
|
|
||||||
</Grid>
|
|
||||||
|
|
||||||
<Box mt={1} mb={2}>
|
<Box mt={1} mb={2}>
|
||||||
<EntityMaskToggle onUpdate={updateDeviceEntity} de={editItem} />
|
<EntityMaskToggle onUpdate={updateDeviceEntity} de={editItem} />
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<Grid container spacing={2}>
|
<Grid container spacing={2}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<TextField
|
||||||
@@ -149,12 +170,14 @@ const CustomizationsDialog = ({
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
{error && (
|
{error && (
|
||||||
<Typography variant="body2" color="error" mt={2}>
|
<Typography variant="body2" color="error" mt={2}>
|
||||||
Error: Check min and max values
|
Error: Check min and max values
|
||||||
</Typography>
|
</Typography>
|
||||||
)}
|
)}
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
|
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useContext, useEffect, useMemo, useState } from 'react';
|
import { memo, useCallback, useContext, useEffect, useMemo, useState } from 'react';
|
||||||
import { IconContext } from 'react-icons/lib';
|
import { IconContext } from 'react-icons/lib';
|
||||||
import { Link } from 'react-router';
|
import { Link } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
@@ -14,6 +14,7 @@ import {
|
|||||||
IconButton,
|
IconButton,
|
||||||
ToggleButton,
|
ToggleButton,
|
||||||
ToggleButtonGroup,
|
ToggleButtonGroup,
|
||||||
|
Tooltip,
|
||||||
Typography
|
Typography
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
|
|
||||||
@@ -44,7 +45,7 @@ import {
|
|||||||
} from './types';
|
} from './types';
|
||||||
import { deviceValueItemValidation } from './validators';
|
import { deviceValueItemValidation } from './validators';
|
||||||
|
|
||||||
const Dashboard = () => {
|
const Dashboard = memo(() => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
@@ -81,7 +82,7 @@ const Dashboard = () => {
|
|||||||
if (!selectedDashboardItem) {
|
if (!selectedDashboardItem) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const id = selectedDashboardItem.id; // this is the parent ID
|
const id = selectedDashboardItem.parentNode.id; // this is the parent ID
|
||||||
await sendDeviceValue({ id, c: devicevalue.c ?? '', v: devicevalue.v })
|
await sendDeviceValue({ id, c: devicevalue.c ?? '', v: devicevalue.v })
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success(LL.WRITE_CMD_SENT());
|
toast.success(LL.WRITE_CMD_SENT());
|
||||||
@@ -132,7 +133,7 @@ const Dashboard = () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const tree = useTree(
|
const tree = useTree(
|
||||||
{ nodes: data.nodes },
|
{ nodes: [...data.nodes] },
|
||||||
{
|
{
|
||||||
onChange: () => {} // not used but needed
|
onChange: () => {} // not used but needed
|
||||||
},
|
},
|
||||||
@@ -163,9 +164,14 @@ const Dashboard = () => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const nodeIds = useMemo(
|
||||||
|
() => data.nodes.map((item: DashboardItem) => item.id),
|
||||||
|
[data.nodes]
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
showAll
|
showAll
|
||||||
? tree.fns.onAddAll(data.nodes.map((item: DashboardItem) => item.id)) // expand tree
|
? tree.fns.onAddAll(nodeIds) // expand tree
|
||||||
: tree.fns.onRemoveAll(); // collapse tree
|
: tree.fns.onRemoveAll(); // collapse tree
|
||||||
}, [parentNodes]);
|
}, [parentNodes]);
|
||||||
|
|
||||||
@@ -195,12 +201,13 @@ const Dashboard = () => {
|
|||||||
[LL]
|
[LL]
|
||||||
);
|
);
|
||||||
|
|
||||||
const showName = (di: DashboardItem) => {
|
const showName = useCallback(
|
||||||
|
(di: DashboardItem) => {
|
||||||
if (di.id < 100) {
|
if (di.id < 100) {
|
||||||
// if its a device (parent node) and has entities
|
// if its a device (parent node) and has entities
|
||||||
if (di.nodes?.length) {
|
if (di.nodes?.length) {
|
||||||
return (
|
return (
|
||||||
<span style={{ fontWeight: 'bold', fontSize: '14px' }}>
|
<span style={{ fontSize: '15px' }}>
|
||||||
<DeviceIcon type_id={di.t ?? 0} />
|
<DeviceIcon type_id={di.t ?? 0} />
|
||||||
{showType(di.n, di.t)}
|
{showType(di.n, di.t)}
|
||||||
<span style={{ color: 'lightblue' }}> ({di.nodes?.length})</span>
|
<span style={{ color: 'lightblue' }}> ({di.nodes?.length})</span>
|
||||||
@@ -212,10 +219,14 @@ const Dashboard = () => {
|
|||||||
return <span>{di.dv.id.slice(2)}</span>;
|
return <span>{di.dv.id.slice(2)}</span>;
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
};
|
},
|
||||||
|
[showType]
|
||||||
|
);
|
||||||
|
|
||||||
const hasMask = (id: string, mask: number) =>
|
const hasMask = useCallback(
|
||||||
(parseInt(id.slice(0, 2), 16) & mask) === mask;
|
(id: string, mask: number) => (parseInt(id.slice(0, 2), 16) & mask) === mask,
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
const editDashboardValue = useCallback(
|
const editDashboardValue = useCallback(
|
||||||
(di: DashboardItem) => {
|
(di: DashboardItem) => {
|
||||||
@@ -237,6 +248,11 @@ const Dashboard = () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const hasFavEntities = useMemo(
|
||||||
|
() => data.nodes.filter((item: DashboardItem) => item.id <= 90).length,
|
||||||
|
[data.nodes]
|
||||||
|
);
|
||||||
|
|
||||||
const renderContent = () => {
|
const renderContent = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return (
|
return (
|
||||||
@@ -244,14 +260,20 @@ const Dashboard = () => {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const hasFavEntities = data.nodes.filter(
|
|
||||||
(item: DashboardItem) => item.id <= 90
|
|
||||||
).length;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{!data.connected && (
|
{!data.connected && (
|
||||||
<MessageBox mb={2} level="error" message={LL.EMS_BUS_WARNING()} />
|
<MessageBox level="error" message={LL.EMS_BUS_WARNING() + '.'}>
|
||||||
|
(
|
||||||
|
<Link
|
||||||
|
target="_blank"
|
||||||
|
to="https://docs.emsesp.org/Troubleshooting#ems-bus-is-not-connecting"
|
||||||
|
style={{ color: 'white' }}
|
||||||
|
>
|
||||||
|
{LL.ONLINE_HELP()}
|
||||||
|
</Link>
|
||||||
|
)
|
||||||
|
</MessageBox>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{data.connected && data.nodes.length > 0 && !hasFavEntities && (
|
{data.connected && data.nodes.length > 0 && !hasFavEntities && (
|
||||||
@@ -271,39 +293,34 @@ const Dashboard = () => {
|
|||||||
</MessageBox>
|
</MessageBox>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{data.nodes.length > 0 && (
|
<Box
|
||||||
<>
|
display="flex"
|
||||||
|
justifyContent="flex-end"
|
||||||
|
flexWrap="nowrap"
|
||||||
|
whiteSpace="nowrap"
|
||||||
|
>
|
||||||
<ToggleButtonGroup
|
<ToggleButtonGroup
|
||||||
color="primary"
|
|
||||||
size="small"
|
size="small"
|
||||||
|
color="primary"
|
||||||
value={showAll}
|
value={showAll}
|
||||||
exclusive
|
exclusive
|
||||||
onChange={handleShowAll}
|
onChange={handleShowAll}
|
||||||
>
|
>
|
||||||
<ButtonTooltip title={LL.ALLVALUES()} arrow>
|
<ButtonTooltip title={LL.ALLVALUES()}>
|
||||||
<ToggleButton value={true}>
|
<ToggleButton value={true}>
|
||||||
<UnfoldMoreIcon sx={{ fontSize: 18 }} />
|
<UnfoldMoreIcon sx={{ fontSize: 18 }} />
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
<ButtonTooltip title={LL.COMPACT()} arrow>
|
<ButtonTooltip title={LL.COMPACT()}>
|
||||||
<ToggleButton value={false}>
|
<ToggleButton value={false}>
|
||||||
<UnfoldLessIcon sx={{ fontSize: 18 }} />
|
<UnfoldLessIcon sx={{ fontSize: 18 }} />
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
</ToggleButtonGroup>
|
</ToggleButtonGroup>
|
||||||
<ButtonTooltip title={LL.DASHBOARD_1()} arrow>
|
</Box>
|
||||||
<HelpOutlineIcon color="primary" sx={{ ml: 1, fontSize: 20 }} />
|
|
||||||
</ButtonTooltip>
|
|
||||||
|
|
||||||
<Box
|
{data.nodes.length > 0 ? (
|
||||||
padding={1}
|
<Box mt={1} justifyContent="center" flexDirection="column">
|
||||||
justifyContent="center"
|
|
||||||
flexDirection="column"
|
|
||||||
sx={{
|
|
||||||
borderRadius: 1,
|
|
||||||
border: '1px solid grey'
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<IconContext.Provider
|
<IconContext.Provider
|
||||||
value={{
|
value={{
|
||||||
color: 'lightblue',
|
color: 'lightblue',
|
||||||
@@ -339,12 +356,12 @@ const Dashboard = () => {
|
|||||||
<Cell>
|
<Cell>
|
||||||
{me.admin &&
|
{me.admin &&
|
||||||
di.dv?.c &&
|
di.dv?.c &&
|
||||||
!hasMask(
|
!hasMask(di.dv.id, DeviceEntityMask.DV_READONLY) && (
|
||||||
di.dv.id,
|
|
||||||
DeviceEntityMask.DV_READONLY
|
|
||||||
) && (
|
|
||||||
<IconButton
|
<IconButton
|
||||||
size="small"
|
size="small"
|
||||||
|
aria-label={
|
||||||
|
LL.CHANGE_VALUE() + ' ' + LL.VALUE(0)
|
||||||
|
}
|
||||||
onClick={() => editDashboardValue(di)}
|
onClick={() => editDashboardValue(di)}
|
||||||
>
|
>
|
||||||
<EditIcon
|
<EditIcon
|
||||||
@@ -369,7 +386,28 @@ const Dashboard = () => {
|
|||||||
</Table>
|
</Table>
|
||||||
</IconContext.Provider>
|
</IconContext.Provider>
|
||||||
</Box>
|
</Box>
|
||||||
</>
|
) : (
|
||||||
|
<Box
|
||||||
|
display="flex"
|
||||||
|
// justifyContent="flex-end"
|
||||||
|
// flexWrap="nowrap"
|
||||||
|
// whiteSpace="nowrap"
|
||||||
|
>
|
||||||
|
<Typography mt={1} color="warning.main" variant="body1">
|
||||||
|
no data
|
||||||
|
</Typography>
|
||||||
|
<Tooltip title={LL.DASHBOARD_1()}>
|
||||||
|
<HelpOutlineIcon
|
||||||
|
sx={{
|
||||||
|
ml: 1,
|
||||||
|
mt: 1,
|
||||||
|
fontSize: 20,
|
||||||
|
verticalAlign: 'middle'
|
||||||
|
}}
|
||||||
|
color="primary"
|
||||||
|
/>
|
||||||
|
</Tooltip>
|
||||||
|
</Box>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
@@ -391,6 +429,6 @@ const Dashboard = () => {
|
|||||||
)}
|
)}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
});
|
||||||
|
|
||||||
export default Dashboard;
|
export default Dashboard;
|
||||||
|
|||||||
@@ -1,13 +1,14 @@
|
|||||||
|
import { memo } from 'react';
|
||||||
import { AiOutlineAlert, AiOutlineControl, AiOutlineGateway } from 'react-icons/ai';
|
import { AiOutlineAlert, AiOutlineControl, AiOutlineGateway } from 'react-icons/ai';
|
||||||
import { CgSmartHomeBoiler } from 'react-icons/cg';
|
import { CgSmartHomeBoiler } from 'react-icons/cg';
|
||||||
import { FaSolarPanel } from 'react-icons/fa';
|
import { FaSolarPanel } from 'react-icons/fa';
|
||||||
import { GiHeatHaze, GiTap } from 'react-icons/gi';
|
import { GiHeatHaze, GiTap } from 'react-icons/gi';
|
||||||
import { MdPlaylistAdd } from 'react-icons/md';
|
|
||||||
import { MdMoreTime } from 'react-icons/md';
|
|
||||||
import {
|
import {
|
||||||
|
MdMoreTime,
|
||||||
MdOutlineDevices,
|
MdOutlineDevices,
|
||||||
MdOutlinePool,
|
MdOutlinePool,
|
||||||
MdOutlineSensors,
|
MdOutlineSensors,
|
||||||
|
MdPlaylistAdd,
|
||||||
MdThermostatAuto
|
MdThermostatAuto
|
||||||
} from 'react-icons/md';
|
} from 'react-icons/md';
|
||||||
import { PiFan, PiGauge } from 'react-icons/pi';
|
import { PiFan, PiGauge } from 'react-icons/pi';
|
||||||
@@ -18,9 +19,10 @@ import type { SvgIconProps } from '@mui/material';
|
|||||||
|
|
||||||
import { DeviceType } from './types';
|
import { DeviceType } from './types';
|
||||||
|
|
||||||
const deviceIconLookup: {
|
const deviceIconLookup: Record<
|
||||||
[key in DeviceType]: React.ComponentType<SvgIconProps> | undefined;
|
DeviceType,
|
||||||
} = {
|
React.ComponentType<SvgIconProps> | null
|
||||||
|
> = {
|
||||||
[DeviceType.TEMPERATURESENSOR]: TiThermometer,
|
[DeviceType.TEMPERATURESENSOR]: TiThermometer,
|
||||||
[DeviceType.ANALOGSENSOR]: PiGauge,
|
[DeviceType.ANALOGSENSOR]: PiGauge,
|
||||||
[DeviceType.BOILER]: CgSmartHomeBoiler,
|
[DeviceType.BOILER]: CgSmartHomeBoiler,
|
||||||
@@ -39,15 +41,19 @@ const deviceIconLookup: {
|
|||||||
[DeviceType.POOL]: MdOutlinePool,
|
[DeviceType.POOL]: MdOutlinePool,
|
||||||
[DeviceType.CUSTOM]: MdPlaylistAdd,
|
[DeviceType.CUSTOM]: MdPlaylistAdd,
|
||||||
[DeviceType.UNKNOWN]: MdOutlineSensors,
|
[DeviceType.UNKNOWN]: MdOutlineSensors,
|
||||||
[DeviceType.SYSTEM]: undefined,
|
[DeviceType.SYSTEM]: null,
|
||||||
[DeviceType.SCHEDULER]: MdMoreTime,
|
[DeviceType.SCHEDULER]: MdMoreTime,
|
||||||
[DeviceType.GENERIC]: MdOutlineSensors,
|
[DeviceType.GENERIC]: MdOutlineSensors,
|
||||||
[DeviceType.VENTILATION]: PiFan
|
[DeviceType.VENTILATION]: PiFan
|
||||||
};
|
};
|
||||||
|
|
||||||
const DeviceIcon = ({ type_id }: { type_id: DeviceType }) => {
|
interface DeviceIconProps {
|
||||||
|
type_id: DeviceType;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DeviceIcon = memo(({ type_id }: DeviceIconProps) => {
|
||||||
const Icon = deviceIconLookup[type_id];
|
const Icon = deviceIconLookup[type_id];
|
||||||
return Icon ? <Icon /> : null;
|
return Icon ? <Icon /> : null;
|
||||||
};
|
});
|
||||||
|
|
||||||
export default DeviceIcon;
|
export default DeviceIcon;
|
||||||
|
|||||||
@@ -1,12 +1,14 @@
|
|||||||
import {
|
import {
|
||||||
|
memo,
|
||||||
useCallback,
|
useCallback,
|
||||||
useContext,
|
useContext,
|
||||||
useEffect,
|
useEffect,
|
||||||
useLayoutEffect,
|
useLayoutEffect,
|
||||||
|
useMemo,
|
||||||
useState
|
useState
|
||||||
} from 'react';
|
} from 'react';
|
||||||
import { IconContext } from 'react-icons';
|
import { IconContext } from 'react-icons';
|
||||||
import { useNavigate } from 'react-router';
|
import { Link, useNavigate } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
||||||
@@ -75,7 +77,7 @@ import { DeviceEntityMask, DeviceType, DeviceValueUOM_s } from './types';
|
|||||||
import type { Device, DeviceValue } from './types';
|
import type { Device, DeviceValue } from './types';
|
||||||
import { deviceValueItemValidation } from './validators';
|
import { deviceValueItemValidation } from './validators';
|
||||||
|
|
||||||
const Devices = () => {
|
const Devices = memo(() => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
@@ -91,7 +93,7 @@ const Devices = () => {
|
|||||||
|
|
||||||
useLayoutTitle(LL.DEVICES());
|
useLayoutTitle(LL.DEVICES());
|
||||||
|
|
||||||
const { data: coreData, send: sendCoreData } = useRequest(() => readCoreData(), {
|
const { data: coreData, send: sendCoreData } = useRequest(readCoreData, {
|
||||||
initialData: {
|
initialData: {
|
||||||
connected: true,
|
connected: true,
|
||||||
devices: []
|
devices: []
|
||||||
@@ -116,32 +118,32 @@ const Devices = () => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
function updateSize() {
|
let raf = 0;
|
||||||
|
const updateSize = () => {
|
||||||
|
cancelAnimationFrame(raf);
|
||||||
|
raf = requestAnimationFrame(() => {
|
||||||
setSize([window.innerWidth, window.innerHeight]);
|
setSize([window.innerWidth, window.innerHeight]);
|
||||||
}
|
});
|
||||||
|
};
|
||||||
window.addEventListener('resize', updateSize);
|
window.addEventListener('resize', updateSize);
|
||||||
updateSize();
|
updateSize();
|
||||||
return () => window.removeEventListener('resize', updateSize);
|
return () => {
|
||||||
|
window.removeEventListener('resize', updateSize);
|
||||||
|
cancelAnimationFrame(raf);
|
||||||
|
};
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const leftOffset = () => {
|
const leftOffset = useCallback(() => {
|
||||||
const devicesWindow = document.getElementById('devices-window');
|
const devicesWindow = document.getElementById('devices-window');
|
||||||
if (!devicesWindow) {
|
if (!devicesWindow) return 0;
|
||||||
return 0;
|
const { left, right } = devicesWindow.getBoundingClientRect();
|
||||||
}
|
if (!left || !right) return 0;
|
||||||
|
|
||||||
const clientRect = devicesWindow.getBoundingClientRect();
|
|
||||||
const left = clientRect.left;
|
|
||||||
const right = clientRect.right;
|
|
||||||
|
|
||||||
if (!left || !right) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return left + (right - left < 400 ? 0 : 200);
|
return left + (right - left < 400 ? 0 : 200);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const common_theme = useTheme({
|
const common_theme = useMemo(
|
||||||
|
() =>
|
||||||
|
useTheme({
|
||||||
BaseRow: `
|
BaseRow: `
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
`,
|
`,
|
||||||
@@ -163,11 +165,21 @@ const Devices = () => {
|
|||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
});
|
}),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
const device_theme = useTheme([
|
const device_theme = useMemo(
|
||||||
|
() =>
|
||||||
|
useTheme([
|
||||||
common_theme,
|
common_theme,
|
||||||
{
|
{
|
||||||
|
BaseRow: `
|
||||||
|
font-size: 15px;
|
||||||
|
.td {
|
||||||
|
height: 28px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
Table: `
|
Table: `
|
||||||
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 130px;
|
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 130px;
|
||||||
`,
|
`,
|
||||||
@@ -176,14 +188,21 @@ const Devices = () => {
|
|||||||
padding: 8px;
|
padding: 8px;
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
font-weight: bold;
|
&:nth-of-type(odd) .td {
|
||||||
|
background-color: #303030;
|
||||||
|
},
|
||||||
&:hover .td {
|
&:hover .td {
|
||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
|
},
|
||||||
`
|
`
|
||||||
}
|
}
|
||||||
]);
|
]),
|
||||||
|
[common_theme]
|
||||||
|
);
|
||||||
|
|
||||||
const data_theme = useTheme([
|
const data_theme = useMemo(
|
||||||
|
() =>
|
||||||
|
useTheme([
|
||||||
common_theme,
|
common_theme,
|
||||||
{
|
{
|
||||||
Table: `
|
Table: `
|
||||||
@@ -225,7 +244,9 @@ const Devices = () => {
|
|||||||
}
|
}
|
||||||
`
|
`
|
||||||
}
|
}
|
||||||
]);
|
]),
|
||||||
|
[common_theme]
|
||||||
|
);
|
||||||
|
|
||||||
const getSortIcon = (state: State, sortKey: unknown) => {
|
const getSortIcon = (state: State, sortKey: unknown) => {
|
||||||
if (state.sortKey === sortKey && state.reverse) {
|
if (state.sortKey === sortKey && state.reverse) {
|
||||||
@@ -238,7 +259,7 @@ const Devices = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const dv_sort = useSort(
|
const dv_sort = useSort(
|
||||||
{ nodes: deviceData.nodes },
|
{ nodes: [...deviceData.nodes] },
|
||||||
{},
|
{},
|
||||||
{
|
{
|
||||||
sortIcon: {
|
sortIcon: {
|
||||||
@@ -268,7 +289,7 @@ const Devices = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const device_select = useRowSelect(
|
const device_select = useRowSelect(
|
||||||
{ nodes: coreData.devices },
|
{ nodes: [...coreData.devices] },
|
||||||
{
|
{
|
||||||
onChange: onSelectChange
|
onChange: onSelectChange
|
||||||
}
|
}
|
||||||
@@ -324,8 +345,10 @@ const Devices = () => {
|
|||||||
return sc;
|
return sc;
|
||||||
};
|
};
|
||||||
|
|
||||||
const hasMask = (id: string, mask: number) =>
|
const hasMask = useCallback(
|
||||||
(parseInt(id.slice(0, 2), 16) & mask) === mask;
|
(id: string, mask: number) => (parseInt(id.slice(0, 2), 16) & mask) === mask,
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
const handleDownloadCsv = () => {
|
const handleDownloadCsv = () => {
|
||||||
const deviceIndex = coreData.devices.findIndex(
|
const deviceIndex = coreData.devices.findIndex(
|
||||||
@@ -510,6 +533,20 @@ const Devices = () => {
|
|||||||
|
|
||||||
const renderCoreData = () => (
|
const renderCoreData = () => (
|
||||||
<>
|
<>
|
||||||
|
{!coreData.connected ? (
|
||||||
|
<MessageBox level="error" message={LL.EMS_BUS_WARNING() + '.'}>
|
||||||
|
(
|
||||||
|
<Link
|
||||||
|
target="_blank"
|
||||||
|
to="https://docs.emsesp.org/Troubleshooting#ems-bus-is-not-connecting"
|
||||||
|
style={{ color: 'white' }}
|
||||||
|
>
|
||||||
|
{LL.ONLINE_HELP()}
|
||||||
|
</Link>
|
||||||
|
)
|
||||||
|
</MessageBox>
|
||||||
|
) : (
|
||||||
|
<Box justifyContent="center" flexDirection="column">
|
||||||
<IconContext.Provider
|
<IconContext.Provider
|
||||||
value={{
|
value={{
|
||||||
color: 'lightblue',
|
color: 'lightblue',
|
||||||
@@ -517,13 +554,8 @@ const Devices = () => {
|
|||||||
style: { verticalAlign: 'middle' }
|
style: { verticalAlign: 'middle' }
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{!coreData.connected && (
|
|
||||||
<MessageBox my={2} level="error" message={LL.EMS_BUS_WARNING()} />
|
|
||||||
)}
|
|
||||||
|
|
||||||
{coreData.connected && (
|
|
||||||
<Table
|
<Table
|
||||||
data={{ nodes: coreData.devices }}
|
data={{ nodes: [...coreData.devices] }}
|
||||||
select={device_select}
|
select={device_select}
|
||||||
theme={device_theme}
|
theme={device_theme}
|
||||||
layout={{ custom: true }}
|
layout={{ custom: true }}
|
||||||
@@ -557,8 +589,9 @@ const Devices = () => {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Table>
|
</Table>
|
||||||
)}
|
|
||||||
</IconContext.Provider>
|
</IconContext.Provider>
|
||||||
|
</Box>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -574,12 +607,13 @@ const Devices = () => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const showDeviceValue = (dv: DeviceValue) => {
|
const showDeviceValue = useCallback((dv: DeviceValue) => {
|
||||||
setSelectedDeviceValue(dv);
|
setSelectedDeviceValue(dv);
|
||||||
setDeviceValueDialogOpen(true);
|
setDeviceValueDialogOpen(true);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const renderNameCell = (dv: DeviceValue) => (
|
const renderNameCell = useCallback(
|
||||||
|
(dv: DeviceValue) => (
|
||||||
<>
|
<>
|
||||||
{dv.id.slice(2)}
|
{dv.id.slice(2)}
|
||||||
{hasMask(dv.id, DeviceEntityMask.DV_FAVORITE) && (
|
{hasMask(dv.id, DeviceEntityMask.DV_FAVORITE) && (
|
||||||
@@ -592,17 +626,22 @@ const Devices = () => {
|
|||||||
<CommentsDisabledOutlinedIcon color="primary" sx={{ fontSize: 12 }} />
|
<CommentsDisabledOutlinedIcon color="primary" sx={{ fontSize: 12 }} />
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
|
),
|
||||||
|
[hasMask]
|
||||||
);
|
);
|
||||||
|
|
||||||
const shown_data = onlyFav
|
const shown_data = useMemo(() => {
|
||||||
? deviceData.nodes.filter(
|
if (onlyFav) {
|
||||||
|
return deviceData.nodes.filter(
|
||||||
(dv: DeviceValue) =>
|
(dv: DeviceValue) =>
|
||||||
hasMask(dv.id, DeviceEntityMask.DV_FAVORITE) &&
|
hasMask(dv.id, DeviceEntityMask.DV_FAVORITE) &&
|
||||||
dv.id.slice(2).toLowerCase().includes(search.toLowerCase())
|
dv.id.slice(2).toLowerCase().includes(search.toLowerCase())
|
||||||
)
|
);
|
||||||
: deviceData.nodes.filter((dv: DeviceValue) =>
|
}
|
||||||
|
return deviceData.nodes.filter((dv: DeviceValue) =>
|
||||||
dv.id.slice(2).toLowerCase().includes(search.toLowerCase())
|
dv.id.slice(2).toLowerCase().includes(search.toLowerCase())
|
||||||
);
|
);
|
||||||
|
}, [deviceData.nodes, onlyFav, search]);
|
||||||
|
|
||||||
const deviceIndex = coreData.devices.findIndex(
|
const deviceIndex = coreData.devices.findIndex(
|
||||||
(d: Device) => d.id === device_select.state.id
|
(d: Device) => d.id === device_select.state.id
|
||||||
@@ -621,7 +660,7 @@ const Devices = () => {
|
|||||||
sx={{
|
sx={{
|
||||||
backgroundColor: 'black',
|
backgroundColor: 'black',
|
||||||
position: 'absolute',
|
position: 'absolute',
|
||||||
left: () => leftOffset(),
|
left: leftOffset,
|
||||||
right: 0,
|
right: 0,
|
||||||
bottom: 0,
|
bottom: 0,
|
||||||
top: 64,
|
top: 64,
|
||||||
@@ -638,7 +677,7 @@ const Devices = () => {
|
|||||||
</Typography>
|
</Typography>
|
||||||
<Grid justifyContent="flex-end">
|
<Grid justifyContent="flex-end">
|
||||||
<ButtonTooltip title={LL.CLOSE()}>
|
<ButtonTooltip title={LL.CLOSE()}>
|
||||||
<IconButton onClick={resetDeviceSelect}>
|
<IconButton onClick={resetDeviceSelect} aria-label={LL.CLOSE()}>
|
||||||
<HighlightOffIcon color="primary" sx={{ fontSize: 18 }} />
|
<HighlightOffIcon color="primary" sx={{ fontSize: 18 }} />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
@@ -650,6 +689,7 @@ const Devices = () => {
|
|||||||
variant="outlined"
|
variant="outlined"
|
||||||
sx={{ width: '22ch' }}
|
sx={{ width: '22ch' }}
|
||||||
placeholder={LL.SEARCH()}
|
placeholder={LL.SEARCH()}
|
||||||
|
aria-label={LL.SEARCH()}
|
||||||
onChange={(event) => {
|
onChange={(event) => {
|
||||||
setSearch(event.target.value);
|
setSearch(event.target.value);
|
||||||
}}
|
}}
|
||||||
@@ -664,19 +704,22 @@ const Devices = () => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ButtonTooltip title={LL.DEVICE_DETAILS()}>
|
<ButtonTooltip title={LL.DEVICE_DETAILS()}>
|
||||||
<IconButton onClick={() => setShowDeviceInfo(true)}>
|
<IconButton
|
||||||
|
onClick={() => setShowDeviceInfo(true)}
|
||||||
|
aria-label={LL.DEVICE_DETAILS()}
|
||||||
|
>
|
||||||
<InfoOutlinedIcon color="primary" sx={{ fontSize: 18 }} />
|
<InfoOutlinedIcon color="primary" sx={{ fontSize: 18 }} />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
{me.admin && (
|
{me.admin && (
|
||||||
<ButtonTooltip title={LL.CUSTOMIZATIONS()}>
|
<ButtonTooltip title={LL.CUSTOMIZATIONS()}>
|
||||||
<IconButton onClick={customize}>
|
<IconButton onClick={customize} aria-label={LL.CUSTOMIZATIONS()}>
|
||||||
<ConstructionIcon color="primary" sx={{ fontSize: 18 }} />
|
<ConstructionIcon color="primary" sx={{ fontSize: 18 }} />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
)}
|
)}
|
||||||
<ButtonTooltip title={LL.EXPORT()}>
|
<ButtonTooltip title={LL.EXPORT()}>
|
||||||
<IconButton onClick={handleDownloadCsv}>
|
<IconButton onClick={handleDownloadCsv} aria-label={LL.EXPORT()}>
|
||||||
<DownloadIcon color="primary" sx={{ fontSize: 18 }} />
|
<DownloadIcon color="primary" sx={{ fontSize: 18 }} />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
@@ -711,7 +754,7 @@ const Devices = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<Table
|
<Table
|
||||||
data={{ nodes: shown_data }}
|
data={{ nodes: Array.from(shown_data) }}
|
||||||
theme={data_theme}
|
theme={data_theme}
|
||||||
sort={dv_sort}
|
sort={dv_sort}
|
||||||
layout={{ custom: true, fixedHeader: true }}
|
layout={{ custom: true, fixedHeader: true }}
|
||||||
@@ -795,6 +838,6 @@ const Devices = () => {
|
|||||||
)}
|
)}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
});
|
||||||
|
|
||||||
export default Devices;
|
export default Devices;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
@@ -52,7 +52,7 @@ const DevicesDialog = ({
|
|||||||
const [editItem, setEditItem] = useState<DeviceValue>(selectedItem);
|
const [editItem, setEditItem] = useState<DeviceValue>(selectedItem);
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const updateFormValue = updateValue(setEditItem);
|
const updateFormValue = useMemo(() => updateValue(setEditItem), [setEditItem]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
@@ -61,11 +61,7 @@ const DevicesDialog = ({
|
|||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const close = () => {
|
const save = useCallback(async () => {
|
||||||
onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
const save = async () => {
|
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, editItem);
|
||||||
@@ -73,9 +69,10 @@ const DevicesDialog = ({
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
}, [validator, editItem, onSave]);
|
||||||
|
|
||||||
const setUom = (uom?: DeviceValueUOM) => {
|
const setUom = useCallback(
|
||||||
|
(uom?: DeviceValueUOM) => {
|
||||||
if (uom === undefined) {
|
if (uom === undefined) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -89,30 +86,49 @@ const DevicesDialog = ({
|
|||||||
default:
|
default:
|
||||||
return DeviceValueUOM_s[uom];
|
return DeviceValueUOM_s[uom];
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
const showHelperText = (dv: DeviceValue) =>
|
const showHelperText = useCallback((dv: DeviceValue) => {
|
||||||
dv.h ? (
|
if (dv.h) return dv.h;
|
||||||
dv.h
|
if (dv.l) return dv.l.join(' | ');
|
||||||
) : dv.l ? (
|
if (dv.m !== undefined && dv.x !== undefined) {
|
||||||
dv.l.join(' | ')
|
return (
|
||||||
) : dv.m !== undefined && dv.x !== undefined ? (
|
|
||||||
<>
|
<>
|
||||||
{dv.m} → {dv.x}
|
{dv.m} → {dv.x}
|
||||||
</>
|
</>
|
||||||
) : undefined;
|
);
|
||||||
|
}
|
||||||
|
return undefined;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const isCommand = useMemo(
|
||||||
|
() => selectedItem.v === '' && selectedItem.c,
|
||||||
|
[selectedItem.v, selectedItem.c]
|
||||||
|
);
|
||||||
|
|
||||||
|
const dialogTitle = useMemo(() => {
|
||||||
|
if (isCommand) return LL.RUN_COMMAND();
|
||||||
|
return writeable ? LL.CHANGE_VALUE() : LL.VALUE(0);
|
||||||
|
}, [isCommand, writeable, LL]);
|
||||||
|
|
||||||
|
const buttonLabel = useMemo(() => {
|
||||||
|
return isCommand ? LL.EXECUTE() : LL.UPDATE();
|
||||||
|
}, [isCommand, LL]);
|
||||||
|
|
||||||
|
const helperText = useMemo(
|
||||||
|
() => showHelperText(editItem),
|
||||||
|
[editItem, showHelperText]
|
||||||
|
);
|
||||||
|
|
||||||
|
const valueLabel = LL.VALUE(0);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={close}>
|
<Dialog sx={dialogStyle} open={open} onClose={onClose}>
|
||||||
<DialogTitle>
|
<DialogTitle>{dialogTitle}</DialogTitle>
|
||||||
{selectedItem.v === '' && selectedItem.c
|
|
||||||
? LL.RUN_COMMAND()
|
|
||||||
: writeable
|
|
||||||
? LL.CHANGE_VALUE()
|
|
||||||
: LL.VALUE(0)}
|
|
||||||
</DialogTitle>
|
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Box color="warning.main" p={0} pl={0} pr={0} mt={0} mb={2}>
|
<Box color="warning.main" mb={2}>
|
||||||
<Typography variant="body2">{editItem.id.slice(2)}</Typography>
|
<Typography variant="body2">{editItem.id.slice(2)}</Typography>
|
||||||
</Box>
|
</Box>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
@@ -120,8 +136,8 @@ const DevicesDialog = ({
|
|||||||
{editItem.l ? (
|
{editItem.l ? (
|
||||||
<TextField
|
<TextField
|
||||||
name="v"
|
name="v"
|
||||||
// label={LL.VALUE(0)}
|
|
||||||
value={editItem.v}
|
value={editItem.v}
|
||||||
|
aria-label={valueLabel}
|
||||||
disabled={!writeable}
|
disabled={!writeable}
|
||||||
sx={{ width: '30ch' }}
|
sx={{ width: '30ch' }}
|
||||||
select
|
select
|
||||||
@@ -137,7 +153,7 @@ const DevicesDialog = ({
|
|||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors || {}}
|
||||||
name="v"
|
name="v"
|
||||||
label={LL.VALUE(0)}
|
label={valueLabel}
|
||||||
value={numberValue(Math.round((editItem.v as number) * 10) / 10)}
|
value={numberValue(Math.round((editItem.v as number) * 10) / 10)}
|
||||||
autoFocus
|
autoFocus
|
||||||
disabled={!writeable}
|
disabled={!writeable}
|
||||||
@@ -161,7 +177,7 @@ const DevicesDialog = ({
|
|||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors || {}}
|
||||||
name="v"
|
name="v"
|
||||||
label={LL.VALUE(0)}
|
label={valueLabel}
|
||||||
value={editItem.v}
|
value={editItem.v}
|
||||||
disabled={!writeable}
|
disabled={!writeable}
|
||||||
sx={{ width: '30ch' }}
|
sx={{ width: '30ch' }}
|
||||||
@@ -170,9 +186,9 @@ const DevicesDialog = ({
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
{writeable && (
|
{writeable && helperText && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<FormHelperText>{showHelperText(editItem)}</FormHelperText>
|
<FormHelperText>{helperText}</FormHelperText>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
@@ -191,7 +207,7 @@ const DevicesDialog = ({
|
|||||||
<Button
|
<Button
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={close}
|
onClick={onClose}
|
||||||
color="secondary"
|
color="secondary"
|
||||||
>
|
>
|
||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
@@ -202,7 +218,7 @@ const DevicesDialog = ({
|
|||||||
onClick={save}
|
onClick={save}
|
||||||
color="primary"
|
color="primary"
|
||||||
>
|
>
|
||||||
{selectedItem.v === '' && selectedItem.c ? LL.EXECUTE() : LL.UPDATE()}
|
{buttonLabel}
|
||||||
</Button>
|
</Button>
|
||||||
{progress && (
|
{progress && (
|
||||||
<CircularProgress
|
<CircularProgress
|
||||||
@@ -217,7 +233,7 @@ const DevicesDialog = ({
|
|||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
) : (
|
) : (
|
||||||
<Button variant="outlined" onClick={close} color="secondary">
|
<Button variant="outlined" onClick={onClose} color="secondary">
|
||||||
{LL.CLOSE()}
|
{LL.CLOSE()}
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { useCallback, useMemo } from 'react';
|
||||||
|
|
||||||
import { ToggleButton, ToggleButtonGroup } from '@mui/material';
|
import { ToggleButton, ToggleButtonGroup } from '@mui/material';
|
||||||
|
|
||||||
import OptionIcon from './OptionIcon';
|
import OptionIcon from './OptionIcon';
|
||||||
@@ -9,92 +11,132 @@ interface EntityMaskToggleProps {
|
|||||||
de: DeviceEntity;
|
de: DeviceEntity;
|
||||||
}
|
}
|
||||||
|
|
||||||
const EntityMaskToggle = ({ onUpdate, de }: EntityMaskToggleProps) => {
|
// Available mask values
|
||||||
const getMaskNumber = (newMask: string[]) => {
|
const MASK_VALUES = [
|
||||||
let new_mask = 0;
|
DeviceEntityMask.DV_WEB_EXCLUDE, // 1
|
||||||
for (const entry of newMask) {
|
DeviceEntityMask.DV_API_MQTT_EXCLUDE, // 2
|
||||||
new_mask |= Number(entry);
|
DeviceEntityMask.DV_READONLY, // 4
|
||||||
}
|
DeviceEntityMask.DV_FAVORITE, // 8
|
||||||
return new_mask;
|
DeviceEntityMask.DV_DELETED // 128
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts an array of mask strings to a bitmask number
|
||||||
|
*/
|
||||||
|
const getMaskNumber = (newMask: string[]): number => {
|
||||||
|
return newMask.reduce((mask, entry) => mask | Number(entry), 0);
|
||||||
};
|
};
|
||||||
|
|
||||||
const getMaskString = (m: number) => {
|
/**
|
||||||
const new_masks: string[] = [];
|
* Converts a bitmask number to an array of mask strings
|
||||||
if ((m & 1) === 1) {
|
*/
|
||||||
new_masks.push('1');
|
const getMaskString = (mask: number): string[] => {
|
||||||
}
|
return MASK_VALUES.filter((value) => (mask & value) === value).map((value) =>
|
||||||
if ((m & 2) === 2) {
|
String(value)
|
||||||
new_masks.push('2');
|
);
|
||||||
}
|
|
||||||
if ((m & 4) === 4) {
|
|
||||||
new_masks.push('4');
|
|
||||||
}
|
|
||||||
if ((m & 8) === 8) {
|
|
||||||
new_masks.push('8');
|
|
||||||
}
|
|
||||||
if ((m & 128) === 128) {
|
|
||||||
new_masks.push('128');
|
|
||||||
}
|
|
||||||
return new_masks;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if a specific mask bit is set
|
||||||
|
*/
|
||||||
|
const hasMask = (mask: number, flag: number): boolean => (mask & flag) === flag;
|
||||||
|
|
||||||
|
const EntityMaskToggle = ({ onUpdate, de }: EntityMaskToggleProps) => {
|
||||||
|
const handleChange = useCallback(
|
||||||
|
(_event: unknown, mask: string[]) => {
|
||||||
|
// Convert selected masks to a number
|
||||||
|
const newMask = getMaskNumber(mask);
|
||||||
|
const updatedDe = { ...de };
|
||||||
|
|
||||||
|
// Apply business logic for mask interactions
|
||||||
|
// If entity has no name and is set to readonly, also exclude from web
|
||||||
|
if (updatedDe.n === '' && hasMask(newMask, DeviceEntityMask.DV_READONLY)) {
|
||||||
|
updatedDe.m = newMask | DeviceEntityMask.DV_WEB_EXCLUDE;
|
||||||
|
} else {
|
||||||
|
updatedDe.m = newMask;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If excluded from web, cannot be favorite
|
||||||
|
if (hasMask(updatedDe.m, DeviceEntityMask.DV_WEB_EXCLUDE)) {
|
||||||
|
updatedDe.m = updatedDe.m & ~DeviceEntityMask.DV_FAVORITE;
|
||||||
|
}
|
||||||
|
|
||||||
|
onUpdate(updatedDe);
|
||||||
|
},
|
||||||
|
[de, onUpdate]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Memoize mask string value
|
||||||
|
const maskStringValue = useMemo(() => getMaskString(de.m), [de.m]);
|
||||||
|
|
||||||
|
// Memoize disabled states
|
||||||
|
const isFavoriteDisabled = useMemo(
|
||||||
|
() =>
|
||||||
|
hasMask(de.m, DeviceEntityMask.DV_WEB_EXCLUDE | DeviceEntityMask.DV_DELETED) ||
|
||||||
|
de.n === undefined,
|
||||||
|
[de.m, de.n]
|
||||||
|
);
|
||||||
|
|
||||||
|
const isReadonlyDisabled = useMemo(
|
||||||
|
() =>
|
||||||
|
!de.w ||
|
||||||
|
hasMask(de.m, DeviceEntityMask.DV_WEB_EXCLUDE | DeviceEntityMask.DV_FAVORITE),
|
||||||
|
[de.w, de.m]
|
||||||
|
);
|
||||||
|
|
||||||
|
const isApiMqttExcludeDisabled = useMemo(
|
||||||
|
() => de.n === '' || hasMask(de.m, DeviceEntityMask.DV_DELETED),
|
||||||
|
[de.n, de.m]
|
||||||
|
);
|
||||||
|
|
||||||
|
const isWebExcludeDisabled = useMemo(
|
||||||
|
() => de.n === undefined || hasMask(de.m, DeviceEntityMask.DV_DELETED),
|
||||||
|
[de.n, de.m]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Memoize mask flag checks
|
||||||
|
const isFavoriteSet = useMemo(
|
||||||
|
() => hasMask(de.m, DeviceEntityMask.DV_FAVORITE),
|
||||||
|
[de.m]
|
||||||
|
);
|
||||||
|
const isReadonlySet = useMemo(
|
||||||
|
() => hasMask(de.m, DeviceEntityMask.DV_READONLY),
|
||||||
|
[de.m]
|
||||||
|
);
|
||||||
|
const isApiMqttExcludeSet = useMemo(
|
||||||
|
() => hasMask(de.m, DeviceEntityMask.DV_API_MQTT_EXCLUDE),
|
||||||
|
[de.m]
|
||||||
|
);
|
||||||
|
const isWebExcludeSet = useMemo(
|
||||||
|
() => hasMask(de.m, DeviceEntityMask.DV_WEB_EXCLUDE),
|
||||||
|
[de.m]
|
||||||
|
);
|
||||||
|
const isDeletedSet = useMemo(
|
||||||
|
() => hasMask(de.m, DeviceEntityMask.DV_DELETED),
|
||||||
|
[de.m]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ToggleButtonGroup
|
<ToggleButtonGroup
|
||||||
size="small"
|
size="small"
|
||||||
color="secondary"
|
color="secondary"
|
||||||
value={getMaskString(de.m)}
|
value={maskStringValue}
|
||||||
onChange={(_event, mask: string[]) => {
|
onChange={handleChange}
|
||||||
de.m = getMaskNumber(mask);
|
|
||||||
if (de.n === '' && de.m & DeviceEntityMask.DV_READONLY) {
|
|
||||||
de.m = de.m | DeviceEntityMask.DV_WEB_EXCLUDE;
|
|
||||||
}
|
|
||||||
if (de.m & DeviceEntityMask.DV_WEB_EXCLUDE) {
|
|
||||||
de.m = de.m & ~DeviceEntityMask.DV_FAVORITE;
|
|
||||||
}
|
|
||||||
onUpdate(de);
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<ToggleButton value="8" disabled={(de.m & 0x81) !== 0 || de.n === undefined}>
|
<ToggleButton value="8" disabled={isFavoriteDisabled}>
|
||||||
<OptionIcon
|
<OptionIcon type="favorite" isSet={isFavoriteSet} />
|
||||||
type="favorite"
|
|
||||||
isSet={
|
|
||||||
(de.m & DeviceEntityMask.DV_FAVORITE) === DeviceEntityMask.DV_FAVORITE
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value="4" disabled={!de.w || (de.m & 0x83) >= 3}>
|
<ToggleButton value="4" disabled={isReadonlyDisabled}>
|
||||||
<OptionIcon
|
<OptionIcon type="readonly" isSet={isReadonlySet} />
|
||||||
type="readonly"
|
|
||||||
isSet={
|
|
||||||
(de.m & DeviceEntityMask.DV_READONLY) === DeviceEntityMask.DV_READONLY
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value="2" disabled={de.n === '' || (de.m & 0x80) !== 0}>
|
<ToggleButton value="2" disabled={isApiMqttExcludeDisabled}>
|
||||||
<OptionIcon
|
<OptionIcon type="api_mqtt_exclude" isSet={isApiMqttExcludeSet} />
|
||||||
type="api_mqtt_exclude"
|
|
||||||
isSet={
|
|
||||||
(de.m & DeviceEntityMask.DV_API_MQTT_EXCLUDE) ===
|
|
||||||
DeviceEntityMask.DV_API_MQTT_EXCLUDE
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value="1" disabled={de.n === undefined || (de.m & 0x80) !== 0}>
|
<ToggleButton value="1" disabled={isWebExcludeDisabled}>
|
||||||
<OptionIcon
|
<OptionIcon type="web_exclude" isSet={isWebExcludeSet} />
|
||||||
type="web_exclude"
|
|
||||||
isSet={
|
|
||||||
(de.m & DeviceEntityMask.DV_WEB_EXCLUDE) ===
|
|
||||||
DeviceEntityMask.DV_WEB_EXCLUDE
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value="128">
|
<ToggleButton value="128">
|
||||||
<OptionIcon
|
<OptionIcon type="deleted" isSet={isDeletedSet} />
|
||||||
type="deleted"
|
|
||||||
isSet={
|
|
||||||
(de.m & DeviceEntityMask.DV_DELETED) === DeviceEntityMask.DV_DELETED
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</ToggleButtonGroup>
|
</ToggleButtonGroup>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { useContext, useState } from 'react';
|
import { memo, useCallback, useContext, useMemo, useState } from 'react';
|
||||||
|
import type { ReactElement } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CommentIcon from '@mui/icons-material/CommentTwoTone';
|
import CommentIcon from '@mui/icons-material/CommentTwoTone';
|
||||||
@@ -19,6 +20,7 @@ import {
|
|||||||
Stack,
|
Stack,
|
||||||
Typography
|
Typography
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
|
import type { SxProps, Theme } from '@mui/material/styles';
|
||||||
|
|
||||||
import { useRequest } from 'alova/client';
|
import { useRequest } from 'alova/client';
|
||||||
import { SectionContent, useLayoutTitle } from 'components';
|
import { SectionContent, useLayoutTitle } from 'components';
|
||||||
@@ -29,26 +31,61 @@ import { saveFile } from 'utils';
|
|||||||
import { API, callAction } from '../../api/app';
|
import { API, callAction } from '../../api/app';
|
||||||
import type { APIcall } from './types';
|
import type { APIcall } from './types';
|
||||||
|
|
||||||
const Help = () => {
|
interface HelpLink {
|
||||||
|
href: string;
|
||||||
|
icon: ReactElement;
|
||||||
|
label: () => string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface CustomSupport {
|
||||||
|
img_url: string | null;
|
||||||
|
html: string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DEFAULT_IMAGE_URL = 'https://emsesp.org/_media/images/installer.jpeg';
|
||||||
|
|
||||||
|
const SUPPORT_BOX_STYLES: SxProps<Theme> = {
|
||||||
|
borderRadius: 3,
|
||||||
|
border: '1px solid lightblue',
|
||||||
|
justifyContent: 'space-evenly',
|
||||||
|
alignItems: 'center'
|
||||||
|
};
|
||||||
|
|
||||||
|
const IMAGE_STYLES: SxProps<Theme> = {
|
||||||
|
maxHeight: { xs: 100, md: 250 }
|
||||||
|
};
|
||||||
|
|
||||||
|
const AVATAR_STYLES: SxProps<Theme> = {
|
||||||
|
bgcolor: '#72caf9'
|
||||||
|
};
|
||||||
|
|
||||||
|
const HelpComponent = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle(LL.HELP());
|
useLayoutTitle(LL.HELP());
|
||||||
|
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
const [customSupportIMG, setCustomSupportIMG] = useState<string | null>(null);
|
const [customSupport, setCustomSupport] = useState<CustomSupport>({
|
||||||
const [customSupportHTML, setCustomSupportHTML] = useState<string | null>(null);
|
img_url: null,
|
||||||
const [notFound, setNotFound] = useState<boolean>(false);
|
html: null
|
||||||
|
});
|
||||||
|
const [imgError, setImgError] = useState<boolean>(false);
|
||||||
|
|
||||||
useRequest(() => callAction({ action: 'getCustomSupport' })).onSuccess((event) => {
|
// Memoize the request method to prevent re-creation on every render
|
||||||
if (event && event.data && Object.keys(event.data).length !== 0) {
|
const getCustomSupportMethod = useMemo(
|
||||||
const data = (event.data as { Support: { img_url?: string; html?: string[] } })
|
() => callAction({ action: 'getCustomSupport' }),
|
||||||
.Support;
|
[]
|
||||||
if (data.img_url) {
|
);
|
||||||
setCustomSupportIMG(data.img_url);
|
|
||||||
}
|
useRequest(getCustomSupportMethod).onSuccess((event) => {
|
||||||
if (data.html) {
|
if (event?.data && Object.keys(event.data).length !== 0) {
|
||||||
setCustomSupportHTML(data.html.join('<br/>'));
|
const { Support } = event.data as {
|
||||||
}
|
Support: { img_url?: string; html?: string[] };
|
||||||
|
};
|
||||||
|
setCustomSupport({
|
||||||
|
img_url: Support.img_url || null,
|
||||||
|
html: Support.html?.join('<br/>') || null
|
||||||
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -63,90 +100,88 @@ const Help = () => {
|
|||||||
toast.error(String(error.error?.message || 'An error occurred'));
|
toast.error(String(error.error?.message || 'An error occurred'));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Optimize API call memoization
|
||||||
|
const apiCall = useMemo(() => ({ device: 'system', cmd: 'info', id: 0 }), []);
|
||||||
|
|
||||||
|
const handleDownloadSystemInfo = useCallback(() => {
|
||||||
|
void sendAPI(apiCall);
|
||||||
|
}, [sendAPI, apiCall]);
|
||||||
|
|
||||||
|
const handleImageError = useCallback(() => {
|
||||||
|
setImgError(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// Memoize help links to prevent recreation on every render
|
||||||
|
const helpLinks: HelpLink[] = useMemo(
|
||||||
|
() => [
|
||||||
|
{
|
||||||
|
href: 'https://emsesp.org',
|
||||||
|
icon: <MenuBookIcon />,
|
||||||
|
label: () => LL.HELP_INFORMATION_1()
|
||||||
|
},
|
||||||
|
{
|
||||||
|
href: 'https://discord.gg/GP9DPSgeJq',
|
||||||
|
icon: <CommentIcon />,
|
||||||
|
label: () => LL.HELP_INFORMATION_2()
|
||||||
|
},
|
||||||
|
{
|
||||||
|
href: 'https://github.com/emsesp/EMS-ESP32/issues/new/choose',
|
||||||
|
icon: <GitHubIcon />,
|
||||||
|
label: () => LL.HELP_INFORMATION_3()
|
||||||
|
}
|
||||||
|
],
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const isAdmin = useMemo(() => me?.admin ?? false, [me?.admin]);
|
||||||
|
|
||||||
|
// Memoize image source computation
|
||||||
|
const imageSrc = useMemo(
|
||||||
|
() =>
|
||||||
|
imgError || !customSupport.img_url ? DEFAULT_IMAGE_URL : customSupport.img_url,
|
||||||
|
[imgError, customSupport.img_url]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{customSupportHTML && (
|
{customSupport.html && (
|
||||||
<Stack
|
<Stack
|
||||||
padding={1}
|
padding={1}
|
||||||
mb={2}
|
mb={2}
|
||||||
direction="row"
|
direction="row"
|
||||||
divider={<Divider orientation="vertical" flexItem />}
|
divider={<Divider orientation="vertical" flexItem />}
|
||||||
sx={{
|
sx={SUPPORT_BOX_STYLES}
|
||||||
borderRadius: 3,
|
|
||||||
border: '2px solid grey',
|
|
||||||
justifyContent: 'space-evenly',
|
|
||||||
alignItems: 'center'
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<Typography variant="subtitle1">
|
<Typography variant="subtitle1">
|
||||||
<div dangerouslySetInnerHTML={{ __html: customSupportHTML }} />
|
<div dangerouslySetInnerHTML={{ __html: customSupport.html }} />
|
||||||
</Typography>
|
</Typography>
|
||||||
<Box
|
<Box
|
||||||
component="img"
|
component="img"
|
||||||
referrerPolicy="no-referrer"
|
referrerPolicy="no-referrer"
|
||||||
sx={{
|
sx={IMAGE_STYLES}
|
||||||
maxHeight: { xs: 100, md: 250 }
|
onError={handleImageError}
|
||||||
}}
|
src={imageSrc}
|
||||||
onError={() => setNotFound(true)}
|
|
||||||
src={
|
|
||||||
notFound
|
|
||||||
? ''
|
|
||||||
: customSupportIMG ||
|
|
||||||
'https://docs.emsesp.org/_media/images/installer.jpeg'
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</Stack>
|
</Stack>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{me.admin && (
|
{isAdmin && (
|
||||||
<List sx={{ borderRadius: 3, border: '2px solid grey' }}>
|
<List>
|
||||||
<ListItem>
|
{helpLinks.map(({ href, icon, label }) => (
|
||||||
|
<ListItem key={href}>
|
||||||
<ListItemButton
|
<ListItemButton
|
||||||
component="a"
|
component="a"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noreferrer"
|
rel="noreferrer"
|
||||||
href="https://docs.emsesp.org"
|
href={href}
|
||||||
>
|
>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: '#72caf9' }}>
|
<Avatar sx={AVATAR_STYLES}>{icon}</Avatar>
|
||||||
<MenuBookIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary={LL.HELP_INFORMATION_1()} />
|
<ListItemText primary={label()} />
|
||||||
</ListItemButton>
|
|
||||||
</ListItem>
|
|
||||||
|
|
||||||
<ListItem>
|
|
||||||
<ListItemButton
|
|
||||||
component="a"
|
|
||||||
target="_blank"
|
|
||||||
rel="noreferrer"
|
|
||||||
href="https://discord.gg/3J3GgnzpyT"
|
|
||||||
>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#72caf9' }}>
|
|
||||||
<CommentIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText primary={LL.HELP_INFORMATION_2()} />
|
|
||||||
</ListItemButton>
|
|
||||||
</ListItem>
|
|
||||||
|
|
||||||
<ListItem>
|
|
||||||
<ListItemButton
|
|
||||||
component="a"
|
|
||||||
target="_blank"
|
|
||||||
rel="noreferrer"
|
|
||||||
href="https://github.com/emsesp/EMS-ESP32/issues/new/choose"
|
|
||||||
>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#72caf9' }}>
|
|
||||||
<GitHubIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText primary={LL.HELP_INFORMATION_3()} />
|
|
||||||
</ListItemButton>
|
</ListItemButton>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
))}
|
||||||
</List>
|
</List>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
@@ -158,7 +193,7 @@ const Help = () => {
|
|||||||
startIcon={<DownloadIcon />}
|
startIcon={<DownloadIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="primary"
|
color="primary"
|
||||||
onClick={() => sendAPI({ device: 'system', cmd: 'info', id: 0 })}
|
onClick={handleDownloadSystemInfo}
|
||||||
>
|
>
|
||||||
{LL.SUPPORT_INFORMATION(0)}
|
{LL.SUPPORT_INFORMATION(0)}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -174,11 +209,14 @@ const Help = () => {
|
|||||||
href="https://emsesp.org"
|
href="https://emsesp.org"
|
||||||
color="primary"
|
color="primary"
|
||||||
>
|
>
|
||||||
{'emsesp.org'}
|
emsesp.org
|
||||||
</Link>
|
</Link>
|
||||||
</Typography>
|
</Typography>
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Memoize the component to prevent unnecessary re-renders
|
||||||
|
const Help = memo(HelpComponent);
|
||||||
|
|
||||||
export default Help;
|
export default Help;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useState } from 'react';
|
import { memo, useCallback, useMemo, useState } from 'react';
|
||||||
import { useBlocker } from 'react-router';
|
import { useBlocker } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
@@ -31,6 +31,19 @@ import { readModules, writeModules } from '../../api/app';
|
|||||||
import ModulesDialog from './ModulesDialog';
|
import ModulesDialog from './ModulesDialog';
|
||||||
import type { ModuleItem } from './types';
|
import type { ModuleItem } from './types';
|
||||||
|
|
||||||
|
const PENDING_COLOR = 'red';
|
||||||
|
const ACTIVATED_COLOR = '#00FF7F';
|
||||||
|
|
||||||
|
const hasModulesChanged = (mi: ModuleItem): boolean =>
|
||||||
|
mi.enabled !== mi.o_enabled || mi.license !== mi.o_license;
|
||||||
|
|
||||||
|
const ColorStatus = memo(({ status }: { status: number }) => {
|
||||||
|
if (status === 1) {
|
||||||
|
return <div style={{ color: PENDING_COLOR }}>Pending Activation</div>;
|
||||||
|
}
|
||||||
|
return <div style={{ color: ACTIVATED_COLOR }}>Activated</div>;
|
||||||
|
});
|
||||||
|
|
||||||
const Modules = () => {
|
const Modules = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [numChanges, setNumChanges] = useState<number>(0);
|
const [numChanges, setNumChanges] = useState<number>(0);
|
||||||
@@ -56,7 +69,9 @@ const Modules = () => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const modules_theme = useTheme({
|
const modules_theme = useTheme(
|
||||||
|
useMemo(
|
||||||
|
() => ({
|
||||||
Table: `
|
Table: `
|
||||||
--data-table-library_grid-template-columns: 48px 180px 120px 100px repeat(1, minmax(160px, 1fr)) 180px;
|
--data-table-library_grid-template-columns: 48px 180px 120px 100px repeat(1, minmax(160px, 1fr)) 180px;
|
||||||
`,
|
`,
|
||||||
@@ -96,43 +111,46 @@ const Modules = () => {
|
|||||||
background-color: #303030;
|
background-color: #303030;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
});
|
}),
|
||||||
|
[]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
const onDialogClose = () => {
|
const onDialogClose = useCallback(() => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const onDialogSave = (updatedItem: ModuleItem) => {
|
const updateModuleItem = useCallback((updatedItem: ModuleItem) => {
|
||||||
|
void updateState(readModules(), (data: ModuleItem[]) => {
|
||||||
|
const new_data = data.map((mi) =>
|
||||||
|
mi.id === updatedItem.id ? { ...mi, ...updatedItem } : mi
|
||||||
|
);
|
||||||
|
setNumChanges(new_data.filter(hasModulesChanged).length);
|
||||||
|
return new_data;
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const onDialogSave = useCallback(
|
||||||
|
(updatedItem: ModuleItem) => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
updateModuleItem(updatedItem);
|
updateModuleItem(updatedItem);
|
||||||
};
|
},
|
||||||
|
[updateModuleItem]
|
||||||
|
);
|
||||||
|
|
||||||
const editModuleItem = useCallback((mi: ModuleItem) => {
|
const editModuleItem = useCallback((mi: ModuleItem) => {
|
||||||
setSelectedModuleItem(mi);
|
setSelectedModuleItem(mi);
|
||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onCancel = async () => {
|
const onCancel = useCallback(async () => {
|
||||||
await fetchModules().then(() => {
|
await fetchModules().then(() => {
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
};
|
}, [fetchModules]);
|
||||||
|
|
||||||
function hasModulesChanged(mi: ModuleItem) {
|
const saveModules = useCallback(async () => {
|
||||||
return mi.enabled !== mi.o_enabled || mi.license !== mi.o_license;
|
try {
|
||||||
}
|
|
||||||
|
|
||||||
const updateModuleItem = (updatedItem: ModuleItem) => {
|
|
||||||
void updateState(readModules(), (data: ModuleItem[]) => {
|
|
||||||
const new_data = data.map((mi) =>
|
|
||||||
mi.id === updatedItem.id ? { ...mi, ...updatedItem } : mi
|
|
||||||
);
|
|
||||||
setNumChanges(new_data.filter((mi) => hasModulesChanged(mi)).length);
|
|
||||||
return new_data;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const saveModules = async () => {
|
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
modules.map((condensed_mi: ModuleItem) =>
|
modules.map((condensed_mi: ModuleItem) =>
|
||||||
updateModules({
|
updateModules({
|
||||||
@@ -141,20 +159,17 @@ const Modules = () => {
|
|||||||
license: condensed_mi.license
|
license: condensed_mi.license
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
)
|
);
|
||||||
.then(() => {
|
|
||||||
toast.success(LL.MODULES_UPDATED());
|
toast.success(LL.MODULES_UPDATED());
|
||||||
})
|
} catch (error) {
|
||||||
.catch((error: Error) => {
|
toast.error(error instanceof Error ? error.message : String(error));
|
||||||
toast.error(error.message);
|
} finally {
|
||||||
})
|
|
||||||
.finally(async () => {
|
|
||||||
await fetchModules();
|
await fetchModules();
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
}
|
||||||
};
|
}, [modules, updateModules, LL, fetchModules]);
|
||||||
|
|
||||||
const renderContent = () => {
|
const content = useMemo(() => {
|
||||||
if (!modules) {
|
if (!modules) {
|
||||||
return (
|
return (
|
||||||
<FormLoader onRetry={fetchModules} errorMessage={error?.message || ''} />
|
<FormLoader onRetry={fetchModules} errorMessage={error?.message || ''} />
|
||||||
@@ -169,13 +184,6 @@ const Modules = () => {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const colorStatus = (status: number) => {
|
|
||||||
if (status === 1) {
|
|
||||||
return <div style={{ color: 'red' }}>Pending Activation</div>;
|
|
||||||
}
|
|
||||||
return <div style={{ color: '#00FF7F' }}>Activated</div>;
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Box mb={2} color="warning.main">
|
<Box mb={2} color="warning.main">
|
||||||
@@ -218,7 +226,9 @@ const Modules = () => {
|
|||||||
<Cell>{mi.author}</Cell>
|
<Cell>{mi.author}</Cell>
|
||||||
<Cell>{mi.version}</Cell>
|
<Cell>{mi.version}</Cell>
|
||||||
<Cell>{mi.message}</Cell>
|
<Cell>{mi.message}</Cell>
|
||||||
<Cell>{colorStatus(mi.status)}</Cell>
|
<Cell>
|
||||||
|
<ColorStatus status={mi.status} />
|
||||||
|
</Cell>
|
||||||
</Row>
|
</Row>
|
||||||
))}
|
))}
|
||||||
</Body>
|
</Body>
|
||||||
@@ -252,12 +262,22 @@ const Modules = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
}, [
|
||||||
|
modules,
|
||||||
|
fetchModules,
|
||||||
|
error,
|
||||||
|
modules_theme,
|
||||||
|
editModuleItem,
|
||||||
|
LL,
|
||||||
|
numChanges,
|
||||||
|
onCancel,
|
||||||
|
saveModules
|
||||||
|
]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{renderContent()}
|
{content}
|
||||||
{selectedModuleItem && (
|
{selectedModuleItem && (
|
||||||
<ModulesDialog
|
<ModulesDialog
|
||||||
open={dialogOpen}
|
open={dialogOpen}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import DoneIcon from '@mui/icons-material/Done';
|
import DoneIcon from '@mui/icons-material/Done';
|
||||||
@@ -37,25 +37,35 @@ const ModulesDialog = ({
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [editItem, setEditItem] = useState<ModuleItem>(selectedItem);
|
const [editItem, setEditItem] = useState<ModuleItem>(selectedItem);
|
||||||
|
|
||||||
const updateFormValue = updateValue(setEditItem);
|
const updateFormValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValue(
|
||||||
|
setEditItem as unknown as React.Dispatch<
|
||||||
|
React.SetStateAction<Record<string, unknown>>
|
||||||
|
>
|
||||||
|
),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Sync form state when dialog opens or selected item changes
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
setEditItem(selectedItem);
|
setEditItem(selectedItem);
|
||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const close = () => {
|
const handleSave = useCallback(() => {
|
||||||
onClose();
|
|
||||||
};
|
|
||||||
|
|
||||||
const save = () => {
|
|
||||||
onSave(editItem);
|
onSave(editItem);
|
||||||
};
|
}, [editItem, onSave]);
|
||||||
|
|
||||||
|
const dialogTitle = useMemo(
|
||||||
|
() => `${LL.EDIT()} ${editItem.key}`,
|
||||||
|
[LL, editItem.key]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} fullWidth maxWidth="xs" open={open} onClose={onClose}>
|
<Dialog sx={dialogStyle} fullWidth maxWidth="xs" open={open} onClose={onClose}>
|
||||||
<DialogTitle>{LL.EDIT() + ' ' + editItem.key}</DialogTitle>
|
<DialogTitle>{dialogTitle}</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
<BlockFormControlLabel
|
<BlockFormControlLabel
|
||||||
@@ -85,7 +95,7 @@ const ModulesDialog = ({
|
|||||||
<Button
|
<Button
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={close}
|
onClick={onClose}
|
||||||
color="secondary"
|
color="secondary"
|
||||||
>
|
>
|
||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
@@ -93,7 +103,7 @@ const ModulesDialog = ({
|
|||||||
<Button
|
<Button
|
||||||
startIcon={<DoneIcon />}
|
startIcon={<DoneIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={save}
|
onClick={handleSave}
|
||||||
color="primary"
|
color="primary"
|
||||||
>
|
>
|
||||||
{LL.UPDATE()}
|
{LL.UPDATE()}
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { memo } from 'react';
|
||||||
|
|
||||||
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
||||||
import DeleteForeverIcon from '@mui/icons-material/DeleteForever';
|
import DeleteForeverIcon from '@mui/icons-material/DeleteForever';
|
||||||
import DeleteOutlineIcon from '@mui/icons-material/DeleteOutline';
|
import DeleteOutlineIcon from '@mui/icons-material/DeleteOutline';
|
||||||
@@ -10,33 +12,39 @@ import VisibilityOffOutlinedIcon from '@mui/icons-material/VisibilityOffOutlined
|
|||||||
import VisibilityOutlinedIcon from '@mui/icons-material/VisibilityOutlined';
|
import VisibilityOutlinedIcon from '@mui/icons-material/VisibilityOutlined';
|
||||||
import type { SvgIconProps } from '@mui/material';
|
import type { SvgIconProps } from '@mui/material';
|
||||||
|
|
||||||
type OptionType =
|
export type OptionType =
|
||||||
| 'deleted'
|
| 'deleted'
|
||||||
| 'readonly'
|
| 'readonly'
|
||||||
| 'web_exclude'
|
| 'web_exclude'
|
||||||
| 'api_mqtt_exclude'
|
| 'api_mqtt_exclude'
|
||||||
| 'favorite';
|
| 'favorite';
|
||||||
|
|
||||||
const OPTION_ICONS: {
|
type IconPair = [
|
||||||
[type in OptionType]: [
|
|
||||||
React.ComponentType<SvgIconProps>,
|
React.ComponentType<SvgIconProps>,
|
||||||
React.ComponentType<SvgIconProps>
|
React.ComponentType<SvgIconProps>
|
||||||
];
|
];
|
||||||
} = {
|
|
||||||
|
const OPTION_ICONS: Record<OptionType, IconPair> = {
|
||||||
deleted: [DeleteForeverIcon, DeleteOutlineIcon],
|
deleted: [DeleteForeverIcon, DeleteOutlineIcon],
|
||||||
readonly: [EditOffOutlinedIcon, EditOutlinedIcon],
|
readonly: [EditOffOutlinedIcon, EditOutlinedIcon],
|
||||||
web_exclude: [VisibilityOffOutlinedIcon, VisibilityOutlinedIcon],
|
web_exclude: [VisibilityOffOutlinedIcon, VisibilityOutlinedIcon],
|
||||||
api_mqtt_exclude: [CommentsDisabledOutlinedIcon, InsertCommentOutlinedIcon],
|
api_mqtt_exclude: [CommentsDisabledOutlinedIcon, InsertCommentOutlinedIcon],
|
||||||
favorite: [StarIcon, StarOutlineIcon]
|
favorite: [StarIcon, StarOutlineIcon]
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
const ICON_SIZE = 16;
|
||||||
|
const ICON_SX = { fontSize: ICON_SIZE, verticalAlign: 'middle' } as const;
|
||||||
|
|
||||||
|
export interface OptionIconProps {
|
||||||
|
readonly type: OptionType;
|
||||||
|
readonly isSet: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const OptionIcon = ({ type, isSet }: OptionIconProps) => {
|
||||||
|
const [SetIcon, UnsetIcon] = OPTION_ICONS[type];
|
||||||
|
const Icon = isSet ? SetIcon : UnsetIcon;
|
||||||
|
|
||||||
|
return <Icon {...(isSet && { color: 'primary' })} sx={ICON_SX} />;
|
||||||
};
|
};
|
||||||
|
|
||||||
const OptionIcon = ({ type, isSet }: { type: OptionType; isSet: boolean }) => {
|
export default memo(OptionIcon);
|
||||||
const Icon = OPTION_ICONS[type][isSet ? 0 : 1];
|
|
||||||
return isSet ? (
|
|
||||||
<Icon color="primary" sx={{ fontSize: 16, verticalAlign: 'middle' }} />
|
|
||||||
) : (
|
|
||||||
<Icon sx={{ fontSize: 16, verticalAlign: 'middle' }} />
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default OptionIcon;
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
import { useBlocker } from 'react-router';
|
import { useBlocker } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
@@ -35,64 +35,31 @@ import { ScheduleFlag } from './types';
|
|||||||
import type { Schedule, ScheduleItem } from './types';
|
import type { Schedule, ScheduleItem } from './types';
|
||||||
import { schedulerItemValidation } from './validators';
|
import { schedulerItemValidation } from './validators';
|
||||||
|
|
||||||
const Scheduler = () => {
|
// Constants
|
||||||
const { LL, locale } = useI18nContext();
|
const INTERVAL_DELAY = 30000; // 30 seconds
|
||||||
const [numChanges, setNumChanges] = useState<number>(0);
|
const MIN_ID = -100;
|
||||||
const blocker = useBlocker(numChanges !== 0);
|
const MAX_ID = 100;
|
||||||
const [selectedScheduleItem, setSelectedScheduleItem] = useState<ScheduleItem>();
|
const ICON_SIZE = 16;
|
||||||
const [dow, setDow] = useState<string[]>([]);
|
const SCHEDULE_FLAG_THRESHOLD = 127;
|
||||||
const [creating, setCreating] = useState<boolean>(false);
|
const FLAG_ALL_DAYS = 127;
|
||||||
const [dialogOpen, setDialogOpen] = useState<boolean>(false);
|
const REFERENCE_YEAR = 2017;
|
||||||
|
const REFERENCE_MONTH = '01';
|
||||||
|
const LOG_2 = Math.log(2);
|
||||||
|
|
||||||
useLayoutTitle(LL.SCHEDULER());
|
// Days of week starting from Monday (1-7)
|
||||||
|
const WEEK_DAYS = [1, 2, 3, 4, 5, 6, 7] as const;
|
||||||
|
|
||||||
const {
|
const DEFAULT_SCHEDULE_ITEM: Omit<ScheduleItem, 'id' | 'o_id'> = {
|
||||||
data: schedule,
|
active: false,
|
||||||
send: fetchSchedule,
|
deleted: false,
|
||||||
error
|
flags: FLAG_ALL_DAYS,
|
||||||
} = useRequest(readSchedule, {
|
time: '',
|
||||||
initialData: []
|
cmd: '',
|
||||||
});
|
value: '',
|
||||||
|
name: ''
|
||||||
|
};
|
||||||
|
|
||||||
const { send: updateSchedule } = useRequest(
|
const scheduleTheme = {
|
||||||
(data: Schedule) => writeSchedule(data),
|
|
||||||
{
|
|
||||||
immediate: false
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
function hasScheduleChanged(si: ScheduleItem) {
|
|
||||||
return (
|
|
||||||
si.id !== si.o_id ||
|
|
||||||
(si.name || '') !== (si.o_name || '') ||
|
|
||||||
si.active !== si.o_active ||
|
|
||||||
si.deleted !== si.o_deleted ||
|
|
||||||
si.flags !== si.o_flags ||
|
|
||||||
si.time !== si.o_time ||
|
|
||||||
si.cmd !== si.o_cmd ||
|
|
||||||
si.value !== si.o_value
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
useInterval(() => {
|
|
||||||
if (numChanges === 0) {
|
|
||||||
void fetchSchedule();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const formatter = new Intl.DateTimeFormat(locale, {
|
|
||||||
weekday: 'short',
|
|
||||||
timeZone: 'UTC'
|
|
||||||
});
|
|
||||||
const days = [1, 2, 3, 4, 5, 6, 7].map((day) => {
|
|
||||||
const dd = day < 10 ? `0${day}` : day;
|
|
||||||
return new Date(`2017-01-${dd}T00:00:00+00:00`);
|
|
||||||
});
|
|
||||||
setDow(days.map((date) => formatter.format(date)));
|
|
||||||
}, [locale]);
|
|
||||||
|
|
||||||
const schedule_theme = useTheme({
|
|
||||||
Table: `
|
Table: `
|
||||||
--data-table-library_grid-template-columns: 36px 210px 100px 192px repeat(1, minmax(100px, 1fr)) 160px;
|
--data-table-library_grid-template-columns: 36px 210px 100px 192px repeat(1, minmax(100px, 1fr)) 160px;
|
||||||
`,
|
`,
|
||||||
@@ -130,9 +97,80 @@ const Scheduler = () => {
|
|||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
|
};
|
||||||
|
|
||||||
|
const scheduleTypeLabels: Record<number, string> = {
|
||||||
|
[ScheduleFlag.SCHEDULE_IMMEDIATE]: 'Immediate',
|
||||||
|
[ScheduleFlag.SCHEDULE_TIMER]: 'Timer',
|
||||||
|
[ScheduleFlag.SCHEDULE_CONDITION]: 'Condition',
|
||||||
|
[ScheduleFlag.SCHEDULE_ONCHANGE]: 'On Change'
|
||||||
|
};
|
||||||
|
|
||||||
|
const Scheduler = () => {
|
||||||
|
const { LL, locale } = useI18nContext();
|
||||||
|
const [numChanges, setNumChanges] = useState<number>(0);
|
||||||
|
const blocker = useBlocker(numChanges !== 0);
|
||||||
|
const [selectedScheduleItem, setSelectedScheduleItem] = useState<ScheduleItem>();
|
||||||
|
const [dow, setDow] = useState<string[]>([]);
|
||||||
|
const [creating, setCreating] = useState<boolean>(false);
|
||||||
|
const [dialogOpen, setDialogOpen] = useState<boolean>(false);
|
||||||
|
|
||||||
|
useLayoutTitle(LL.SCHEDULER());
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: schedule,
|
||||||
|
send: fetchSchedule,
|
||||||
|
error
|
||||||
|
} = useRequest(readSchedule, {
|
||||||
|
initialData: []
|
||||||
});
|
});
|
||||||
|
|
||||||
const saveSchedule = async () => {
|
const { send: updateSchedule } = useRequest(
|
||||||
|
(data: Schedule) => writeSchedule(data),
|
||||||
|
{
|
||||||
|
immediate: false
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const hasScheduleChanged = useCallback((si: ScheduleItem) => {
|
||||||
|
return (
|
||||||
|
si.id !== si.o_id ||
|
||||||
|
(si.name || '') !== (si.o_name || '') ||
|
||||||
|
si.active !== si.o_active ||
|
||||||
|
si.deleted !== si.o_deleted ||
|
||||||
|
si.flags !== si.o_flags ||
|
||||||
|
si.time !== si.o_time ||
|
||||||
|
si.cmd !== si.o_cmd ||
|
||||||
|
si.value !== si.o_value
|
||||||
|
);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const intervalCallback = useCallback(() => {
|
||||||
|
if (numChanges === 0) {
|
||||||
|
void fetchSchedule();
|
||||||
|
}
|
||||||
|
}, [numChanges, fetchSchedule]);
|
||||||
|
|
||||||
|
useInterval(intervalCallback, INTERVAL_DELAY);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const formatter = new Intl.DateTimeFormat(locale, {
|
||||||
|
weekday: 'short',
|
||||||
|
timeZone: 'UTC'
|
||||||
|
});
|
||||||
|
const days = WEEK_DAYS.map((day) => {
|
||||||
|
const dayStr = String(day).padStart(2, '0');
|
||||||
|
return new Date(
|
||||||
|
`${REFERENCE_YEAR}-${REFERENCE_MONTH}-${dayStr}T00:00:00+00:00`
|
||||||
|
);
|
||||||
|
});
|
||||||
|
setDow(days.map((date) => formatter.format(date)));
|
||||||
|
}, [locale]);
|
||||||
|
|
||||||
|
const schedule_theme = useTheme(scheduleTheme);
|
||||||
|
|
||||||
|
const saveSchedule = useCallback(async () => {
|
||||||
|
try {
|
||||||
await updateSchedule({
|
await updateSchedule({
|
||||||
schedule: schedule
|
schedule: schedule
|
||||||
.filter((si: ScheduleItem) => !si.deleted)
|
.filter((si: ScheduleItem) => !si.deleted)
|
||||||
@@ -145,18 +183,16 @@ const Scheduler = () => {
|
|||||||
value: condensed_si.value,
|
value: condensed_si.value,
|
||||||
name: condensed_si.name
|
name: condensed_si.name
|
||||||
}))
|
}))
|
||||||
})
|
});
|
||||||
.then(() => {
|
|
||||||
toast.success(LL.SCHEDULE_UPDATED());
|
toast.success(LL.SCHEDULE_UPDATED());
|
||||||
})
|
} catch (error: unknown) {
|
||||||
.catch((error: Error) => {
|
const message = error instanceof Error ? error.message : String(error);
|
||||||
toast.error(error.message);
|
toast.error(message);
|
||||||
})
|
} finally {
|
||||||
.finally(async () => {
|
|
||||||
await fetchSchedule();
|
await fetchSchedule();
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
}
|
||||||
};
|
}, [LL, schedule, updateSchedule, fetchSchedule]);
|
||||||
|
|
||||||
const editScheduleItem = useCallback((si: ScheduleItem) => {
|
const editScheduleItem = useCallback((si: ScheduleItem) => {
|
||||||
setCreating(false);
|
setCreating(false);
|
||||||
@@ -167,24 +203,22 @@ const Scheduler = () => {
|
|||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onDialogClose = () => {
|
const onDialogClose = useCallback(() => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const onDialogCancel = async () => {
|
const onDialogCancel = useCallback(async () => {
|
||||||
await fetchSchedule().then(() => {
|
await fetchSchedule().then(() => {
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
};
|
}, [fetchSchedule]);
|
||||||
|
|
||||||
const onDialogSave = (updatedItem: ScheduleItem) => {
|
const onDialogSave = useCallback(
|
||||||
|
(updatedItem: ScheduleItem) => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
void updateState(readSchedule(), (data: ScheduleItem[]) => {
|
void updateState(readSchedule(), (data: ScheduleItem[]) => {
|
||||||
const new_data = creating
|
const new_data = creating
|
||||||
? [
|
? [...data, updatedItem]
|
||||||
...data.filter((si) => creating || si.o_id !== updatedItem.o_id),
|
|
||||||
updatedItem
|
|
||||||
]
|
|
||||||
: data.map((si) =>
|
: data.map((si) =>
|
||||||
si.id === updatedItem.id ? { ...si, ...updatedItem } : si
|
si.id === updatedItem.id ? { ...si, ...updatedItem } : si
|
||||||
);
|
);
|
||||||
@@ -193,69 +227,69 @@ const Scheduler = () => {
|
|||||||
|
|
||||||
return new_data;
|
return new_data;
|
||||||
});
|
});
|
||||||
};
|
},
|
||||||
|
[creating, hasScheduleChanged]
|
||||||
|
);
|
||||||
|
|
||||||
const addScheduleItem = () => {
|
const addScheduleItem = useCallback(() => {
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
setSelectedScheduleItem({
|
const newItem: ScheduleItem = {
|
||||||
id: Math.floor(Math.random() * (Math.floor(200) - 100) + 100),
|
id: Math.floor(Math.random() * (MAX_ID - MIN_ID) + MIN_ID),
|
||||||
active: false,
|
...DEFAULT_SCHEDULE_ITEM
|
||||||
deleted: false,
|
|
||||||
flags: ScheduleFlag.SCHEDULE_DAY,
|
|
||||||
time: '',
|
|
||||||
cmd: '',
|
|
||||||
value: '',
|
|
||||||
name: ''
|
|
||||||
});
|
|
||||||
setDialogOpen(true);
|
|
||||||
};
|
};
|
||||||
|
setSelectedScheduleItem(newItem);
|
||||||
|
setDialogOpen(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const renderSchedule = () => {
|
const filteredAndSortedSchedule = useMemo(
|
||||||
|
() =>
|
||||||
|
schedule
|
||||||
|
.filter((si: ScheduleItem) => !si.deleted)
|
||||||
|
.sort((a: ScheduleItem, b: ScheduleItem) => a.flags - b.flags),
|
||||||
|
[schedule]
|
||||||
|
);
|
||||||
|
|
||||||
|
const dayBox = useCallback(
|
||||||
|
(si: ScheduleItem, flag: number) => {
|
||||||
|
const dayIndex = Math.log(flag) / LOG_2;
|
||||||
|
const isActive = (si.flags & flag) === flag;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Box>
|
||||||
|
<Typography sx={{ fontSize: 11 }} color={isActive ? 'primary' : 'grey'}>
|
||||||
|
{dow[dayIndex]}
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<Divider orientation="vertical" flexItem />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[dow]
|
||||||
|
);
|
||||||
|
|
||||||
|
const scheduleType = useCallback((si: ScheduleItem) => {
|
||||||
|
const label = scheduleTypeLabels[si.flags];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Box>
|
||||||
|
<Typography sx={{ fontSize: 11 }} color="primary">
|
||||||
|
{label || ''}
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const renderSchedule = useCallback(() => {
|
||||||
if (!schedule) {
|
if (!schedule) {
|
||||||
return (
|
return (
|
||||||
<FormLoader onRetry={fetchSchedule} errorMessage={error?.message || ''} />
|
<FormLoader onRetry={fetchSchedule} errorMessage={error?.message || ''} />
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const dayBox = (si: ScheduleItem, flag: number) => (
|
|
||||||
<>
|
|
||||||
<Box>
|
|
||||||
<Typography
|
|
||||||
sx={{ fontSize: 11 }}
|
|
||||||
color={(si.flags & flag) === flag ? 'primary' : 'grey'}
|
|
||||||
>
|
|
||||||
{dow[Math.log(flag) / Math.log(2)]}
|
|
||||||
</Typography>
|
|
||||||
</Box>
|
|
||||||
<Divider orientation="vertical" flexItem />
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
|
|
||||||
const scheduleType = (si: ScheduleItem) => (
|
|
||||||
<Box>
|
|
||||||
<Typography sx={{ fontSize: 11 }} color="primary">
|
|
||||||
{si.flags === ScheduleFlag.SCHEDULE_IMMEDIATE ? (
|
|
||||||
<>Immediate</>
|
|
||||||
) : si.flags === ScheduleFlag.SCHEDULE_TIMER ? (
|
|
||||||
<>Timer</>
|
|
||||||
) : si.flags === ScheduleFlag.SCHEDULE_CONDITION ? (
|
|
||||||
<>Condition</>
|
|
||||||
) : si.flags === ScheduleFlag.SCHEDULE_ONCHANGE ? (
|
|
||||||
<>On Change</>
|
|
||||||
) : (
|
|
||||||
<></>
|
|
||||||
)}
|
|
||||||
</Typography>
|
|
||||||
</Box>
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Table
|
<Table
|
||||||
data={{
|
data={{ nodes: filteredAndSortedSchedule }}
|
||||||
nodes: schedule
|
|
||||||
.filter((si: ScheduleItem) => !si.deleted)
|
|
||||||
.sort((a: ScheduleItem, b: ScheduleItem) => a.flags - b.flags)
|
|
||||||
}}
|
|
||||||
theme={schedule_theme}
|
theme={schedule_theme}
|
||||||
layout={{ custom: true }}
|
layout={{ custom: true }}
|
||||||
>
|
>
|
||||||
@@ -275,22 +309,15 @@ const Scheduler = () => {
|
|||||||
{tableList.map((si: ScheduleItem) => (
|
{tableList.map((si: ScheduleItem) => (
|
||||||
<Row key={si.id} item={si} onClick={() => editScheduleItem(si)}>
|
<Row key={si.id} item={si} onClick={() => editScheduleItem(si)}>
|
||||||
<Cell stiff>
|
<Cell stiff>
|
||||||
{si.active ? (
|
|
||||||
<CircleIcon
|
<CircleIcon
|
||||||
color="success"
|
color={si.active ? 'success' : 'error'}
|
||||||
sx={{ fontSize: 16, verticalAlign: 'middle' }}
|
sx={{ fontSize: ICON_SIZE, verticalAlign: 'middle' }}
|
||||||
/>
|
/>
|
||||||
) : (
|
|
||||||
<CircleIcon
|
|
||||||
color="error"
|
|
||||||
sx={{ fontSize: 16, verticalAlign: 'middle' }}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell stiff>
|
<Cell stiff>
|
||||||
<Stack spacing={0.5} direction="row">
|
<Stack spacing={0.5} direction="row">
|
||||||
<Divider orientation="vertical" flexItem />
|
<Divider orientation="vertical" flexItem />
|
||||||
{si.flags > 127 ? (
|
{si.flags > SCHEDULE_FLAG_THRESHOLD ? (
|
||||||
scheduleType(si)
|
scheduleType(si)
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
@@ -316,7 +343,17 @@ const Scheduler = () => {
|
|||||||
)}
|
)}
|
||||||
</Table>
|
</Table>
|
||||||
);
|
);
|
||||||
};
|
}, [
|
||||||
|
schedule,
|
||||||
|
error,
|
||||||
|
fetchSchedule,
|
||||||
|
filteredAndSortedSchedule,
|
||||||
|
schedule_theme,
|
||||||
|
editScheduleItem,
|
||||||
|
LL,
|
||||||
|
dayBox,
|
||||||
|
scheduleType
|
||||||
|
]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
@@ -338,7 +375,7 @@ const Scheduler = () => {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Box mt={1} display="flex" flexWrap="wrap">
|
<Box display="flex" flexWrap="wrap">
|
||||||
<Box flexGrow={1}>
|
<Box flexGrow={1}>
|
||||||
{numChanges !== 0 && (
|
{numChanges !== 0 && (
|
||||||
<ButtonRow>
|
<ButtonRow>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import AddIcon from '@mui/icons-material/Add';
|
import AddIcon from '@mui/icons-material/Add';
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -31,6 +31,35 @@ import { validate } from 'validators';
|
|||||||
import { ScheduleFlag } from './types';
|
import { ScheduleFlag } from './types';
|
||||||
import type { ScheduleItem } from './types';
|
import type { ScheduleItem } from './types';
|
||||||
|
|
||||||
|
// Constants
|
||||||
|
const FLAG_MASK_127 = 127;
|
||||||
|
const SCHEDULE_TYPE_THRESHOLD = 127;
|
||||||
|
const FLAG_ALL_DAYS = 127;
|
||||||
|
const DEFAULT_TIME = '00:00';
|
||||||
|
const TYPOGRAPHY_FONT_SIZE = 10;
|
||||||
|
|
||||||
|
// Day of week flag configuration (static, defined outside component)
|
||||||
|
const DAY_FLAGS = [
|
||||||
|
{ value: '2', flag: ScheduleFlag.SCHEDULE_MON },
|
||||||
|
{ value: '4', flag: ScheduleFlag.SCHEDULE_TUE },
|
||||||
|
{ value: '8', flag: ScheduleFlag.SCHEDULE_WED },
|
||||||
|
{ value: '16', flag: ScheduleFlag.SCHEDULE_THU },
|
||||||
|
{ value: '32', flag: ScheduleFlag.SCHEDULE_FRI },
|
||||||
|
{ value: '64', flag: ScheduleFlag.SCHEDULE_SAT },
|
||||||
|
{ value: '1', flag: ScheduleFlag.SCHEDULE_SUN }
|
||||||
|
] as const;
|
||||||
|
|
||||||
|
// Day of week flag values array (static)
|
||||||
|
const FLAG_VALUES = [
|
||||||
|
ScheduleFlag.SCHEDULE_SUN,
|
||||||
|
ScheduleFlag.SCHEDULE_MON,
|
||||||
|
ScheduleFlag.SCHEDULE_TUE,
|
||||||
|
ScheduleFlag.SCHEDULE_WED,
|
||||||
|
ScheduleFlag.SCHEDULE_THU,
|
||||||
|
ScheduleFlag.SCHEDULE_FRI,
|
||||||
|
ScheduleFlag.SCHEDULE_SAT
|
||||||
|
] as const;
|
||||||
|
|
||||||
interface SchedulerDialogProps {
|
interface SchedulerDialogProps {
|
||||||
open: boolean;
|
open: boolean;
|
||||||
creating: boolean;
|
creating: boolean;
|
||||||
@@ -53,110 +82,164 @@ const SchedulerDialog = ({
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [editItem, setEditItem] = useState<ScheduleItem>(selectedItem);
|
const [editItem, setEditItem] = useState<ScheduleItem>(selectedItem);
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const [scheduleType, setScheduleType] = useState<ScheduleFlag>();
|
const [scheduleType, setScheduleType] = useState<ScheduleFlag>();
|
||||||
|
|
||||||
const updateFormValue = updateValue(setEditItem);
|
const updateFormValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValue(
|
||||||
|
setEditItem as unknown as React.Dispatch<
|
||||||
|
React.SetStateAction<Record<string, unknown>>
|
||||||
|
>
|
||||||
|
),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
setEditItem(selectedItem);
|
setEditItem(selectedItem);
|
||||||
// set the flags based on type when page is loaded...
|
// Set the flags based on type when page is loaded:
|
||||||
// 0-127 is day schedule
|
// 0-127 is day schedule
|
||||||
// 128 is timer
|
// 128 is timer
|
||||||
// 129 is on change
|
// 129 is on change
|
||||||
// 130 is on condition
|
// 130 is on condition
|
||||||
// 132 is immediate
|
// 132 is immediate
|
||||||
setScheduleType(
|
setScheduleType(
|
||||||
selectedItem.flags < 128 ? ScheduleFlag.SCHEDULE_DAY : selectedItem.flags
|
selectedItem.flags <= SCHEDULE_TYPE_THRESHOLD
|
||||||
|
? ScheduleFlag.SCHEDULE_DAY
|
||||||
|
: selectedItem.flags
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const save = async () => {
|
// Helper function to handle save operations
|
||||||
|
const handleSave = useCallback(
|
||||||
|
async (itemToSave: ScheduleItem) => {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, itemToSave);
|
||||||
onSave(editItem);
|
onSave(itemToSave);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[validator, onSave]
|
||||||
const saveandactivate = async () => {
|
|
||||||
editItem.active = true;
|
|
||||||
try {
|
|
||||||
setFieldErrors(undefined);
|
|
||||||
await validate(validator, editItem);
|
|
||||||
onSave(editItem);
|
|
||||||
} catch (error) {
|
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const remove = () => {
|
|
||||||
editItem.deleted = true;
|
|
||||||
onSave(editItem);
|
|
||||||
};
|
|
||||||
|
|
||||||
const getFlagDOWnumber = (newFlag: string[]) => {
|
|
||||||
let new_flag = 0;
|
|
||||||
for (const entry of newFlag) {
|
|
||||||
new_flag |= Number(entry);
|
|
||||||
}
|
|
||||||
return new_flag & 127;
|
|
||||||
};
|
|
||||||
|
|
||||||
const getFlagDOWstring = (f: number) => {
|
|
||||||
const new_flags: string[] = [];
|
|
||||||
if ((f & 129) === 1) {
|
|
||||||
new_flags.push('1');
|
|
||||||
}
|
|
||||||
if ((f & 130) === 2) {
|
|
||||||
new_flags.push('2');
|
|
||||||
}
|
|
||||||
if ((f & 4) === 4) {
|
|
||||||
new_flags.push('4');
|
|
||||||
}
|
|
||||||
if ((f & 8) === 8) {
|
|
||||||
new_flags.push('8');
|
|
||||||
}
|
|
||||||
if ((f & 16) === 16) {
|
|
||||||
new_flags.push('16');
|
|
||||||
}
|
|
||||||
if ((f & 32) === 32) {
|
|
||||||
new_flags.push('32');
|
|
||||||
}
|
|
||||||
if ((f & 64) === 64) {
|
|
||||||
new_flags.push('64');
|
|
||||||
}
|
|
||||||
|
|
||||||
return new_flags;
|
|
||||||
};
|
|
||||||
|
|
||||||
const showDOW = (si: ScheduleItem, flag: number) => (
|
|
||||||
<Typography
|
|
||||||
sx={{ fontSize: 10 }}
|
|
||||||
color={(si.flags & flag) === flag ? 'primary' : 'grey'}
|
|
||||||
>
|
|
||||||
{dow[Math.log(flag) / Math.log(2)]}
|
|
||||||
</Typography>
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const handleClose = (
|
const save = useCallback(async () => {
|
||||||
_event: React.SyntheticEvent,
|
await handleSave(editItem);
|
||||||
reason: 'backdropClick' | 'escapeKeyDown'
|
}, [editItem, handleSave]);
|
||||||
) => {
|
|
||||||
|
const saveandactivate = useCallback(async () => {
|
||||||
|
await handleSave({ ...editItem, active: true });
|
||||||
|
}, [editItem, handleSave]);
|
||||||
|
|
||||||
|
const remove = useCallback(() => {
|
||||||
|
onSave({ ...editItem, deleted: true });
|
||||||
|
}, [editItem, onSave]);
|
||||||
|
|
||||||
|
// Optimize DOW flag conversion
|
||||||
|
const getFlagDOWnumber = useCallback((flags: string[]) => {
|
||||||
|
return flags.reduce((acc, flag) => acc | Number(flag), 0) & FLAG_MASK_127;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const getFlagDOWstring = useCallback((f: number) => {
|
||||||
|
return FLAG_VALUES.filter((flag) => (f & flag) === flag).map((flag) =>
|
||||||
|
String(flag)
|
||||||
|
);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// Day of week display component
|
||||||
|
const DayOfWeekButton = useCallback(
|
||||||
|
(flag: number) => {
|
||||||
|
const dayIndex = Math.log2(flag);
|
||||||
|
const isSelected = (editItem.flags & flag) === flag;
|
||||||
|
return (
|
||||||
|
<Typography
|
||||||
|
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
||||||
|
color={isSelected ? 'primary' : 'grey'}
|
||||||
|
>
|
||||||
|
{dow[dayIndex]}
|
||||||
|
</Typography>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[editItem.flags, dow]
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleClose = useCallback(
|
||||||
|
(_event: React.SyntheticEvent, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
if (reason !== 'backdropClick') {
|
if (reason !== 'backdropClick') {
|
||||||
onClose();
|
onClose();
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[onClose]
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleScheduleTypeChange = useCallback(
|
||||||
|
(_event: React.SyntheticEvent<HTMLElement>, flag: ScheduleFlag | null) => {
|
||||||
|
if (flag !== null) {
|
||||||
|
setFieldErrors(undefined); // clear any validation errors
|
||||||
|
setScheduleType(flag);
|
||||||
|
// wipe the time field when changing the schedule type
|
||||||
|
// set the flags based on type
|
||||||
|
const newFlags = flag === ScheduleFlag.SCHEDULE_DAY ? FLAG_ALL_DAYS : flag;
|
||||||
|
setEditItem((prev) => ({ ...prev, time: '', flags: newFlags }));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleDOWChange = useCallback(
|
||||||
|
(_event: React.SyntheticEvent<HTMLElement>, flags: string[]) => {
|
||||||
|
const newFlags =
|
||||||
|
getFlagDOWnumber(flags) === 0 ? FLAG_ALL_DAYS : getFlagDOWnumber(flags);
|
||||||
|
setEditItem((prev) => ({ ...prev, flags: newFlags }));
|
||||||
|
},
|
||||||
|
[getFlagDOWnumber]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Memoize derived values
|
||||||
|
const isDaySchedule = useMemo(
|
||||||
|
() => scheduleType === ScheduleFlag.SCHEDULE_DAY,
|
||||||
|
[scheduleType]
|
||||||
|
);
|
||||||
|
const isTimerSchedule = useMemo(
|
||||||
|
() => scheduleType === ScheduleFlag.SCHEDULE_TIMER,
|
||||||
|
[scheduleType]
|
||||||
|
);
|
||||||
|
const isImmediateSchedule = useMemo(
|
||||||
|
() => scheduleType === ScheduleFlag.SCHEDULE_IMMEDIATE,
|
||||||
|
[scheduleType]
|
||||||
|
);
|
||||||
|
const needsTimeField = useMemo(
|
||||||
|
() => isDaySchedule || isTimerSchedule,
|
||||||
|
[isDaySchedule, isTimerSchedule]
|
||||||
|
);
|
||||||
|
|
||||||
|
const dowFlags = useMemo(
|
||||||
|
() => getFlagDOWstring(editItem.flags),
|
||||||
|
[editItem.flags, getFlagDOWstring]
|
||||||
|
);
|
||||||
|
|
||||||
|
const timeFieldValue = useMemo(() => {
|
||||||
|
if (needsTimeField) {
|
||||||
|
return editItem.time === '' ? DEFAULT_TIME : editItem.time;
|
||||||
|
}
|
||||||
|
return editItem.time === DEFAULT_TIME ? '' : editItem.time;
|
||||||
|
}, [editItem.time, needsTimeField]);
|
||||||
|
|
||||||
|
const timeFieldLabel = useMemo(() => {
|
||||||
|
if (scheduleType === ScheduleFlag.SCHEDULE_TIMER) return LL.TIMER(1);
|
||||||
|
if (scheduleType === ScheduleFlag.SCHEDULE_CONDITION) return LL.CONDITION();
|
||||||
|
if (scheduleType === ScheduleFlag.SCHEDULE_ONCHANGE) return LL.ONCHANGE();
|
||||||
|
if (scheduleType === ScheduleFlag.SCHEDULE_IMMEDIATE) return LL.IMMEDIATE();
|
||||||
|
return LL.TIME(1);
|
||||||
|
}, [scheduleType, LL]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
<DialogTitle>
|
<DialogTitle>
|
||||||
{creating ? LL.ADD(1) + ' ' + LL.NEW(0) : LL.EDIT()}
|
{creating ? `${LL.ADD(1)} ${LL.NEW(0)}` : LL.EDIT()}
|
||||||
{LL.SCHEDULE(1)}
|
{LL.SCHEDULE(1)}
|
||||||
</DialogTitle>
|
</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
@@ -166,47 +249,27 @@ const SchedulerDialog = ({
|
|||||||
value={scheduleType}
|
value={scheduleType}
|
||||||
exclusive
|
exclusive
|
||||||
disabled={!creating}
|
disabled={!creating}
|
||||||
onChange={(_event, flag: ScheduleFlag) => {
|
onChange={handleScheduleTypeChange}
|
||||||
if (flag !== null) {
|
|
||||||
setFieldErrors(undefined); // clear any validation errors
|
|
||||||
setScheduleType(flag);
|
|
||||||
// wipe the time field when changing the schedule type
|
|
||||||
setEditItem({ ...editItem, time: '' });
|
|
||||||
// set the flags based on type
|
|
||||||
// 0-127 is day schedule
|
|
||||||
// 128 is timer
|
|
||||||
// 129 is on change
|
|
||||||
// 130 is on condition
|
|
||||||
// 132 is immediate
|
|
||||||
setEditItem(
|
|
||||||
flag === ScheduleFlag.SCHEDULE_DAY
|
|
||||||
? { ...editItem, flags: 0 }
|
|
||||||
: { ...editItem, flags: flag }
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_DAY}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_DAY}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: 10 }}
|
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
||||||
color={scheduleType === ScheduleFlag.SCHEDULE_DAY ? 'primary' : 'grey'}
|
color={isDaySchedule ? 'primary' : 'grey'}
|
||||||
>
|
>
|
||||||
{LL.SCHEDULE(0)}
|
{LL.SCHEDULE(0)}
|
||||||
</Typography>
|
</Typography>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_TIMER}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_TIMER}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: 10 }}
|
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
||||||
color={
|
color={isTimerSchedule ? 'primary' : 'grey'}
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_TIMER ? 'primary' : 'grey'
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
{LL.TIMER(0)}
|
{LL.TIMER(0)}
|
||||||
</Typography>
|
</Typography>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_ONCHANGE}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_ONCHANGE}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: 10 }}
|
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
||||||
color={
|
color={
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_ONCHANGE ? 'primary' : 'grey'
|
scheduleType === ScheduleFlag.SCHEDULE_ONCHANGE ? 'primary' : 'grey'
|
||||||
}
|
}
|
||||||
@@ -216,7 +279,7 @@ const SchedulerDialog = ({
|
|||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_CONDITION}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_CONDITION}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: 10 }}
|
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
||||||
color={
|
color={
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_CONDITION ? 'primary' : 'grey'
|
scheduleType === ScheduleFlag.SCHEDULE_CONDITION ? 'primary' : 'grey'
|
||||||
}
|
}
|
||||||
@@ -226,50 +289,30 @@ const SchedulerDialog = ({
|
|||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_IMMEDIATE}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_IMMEDIATE}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: 10 }}
|
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
||||||
color={
|
color={isImmediateSchedule ? 'primary' : 'grey'}
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_IMMEDIATE ? 'primary' : 'grey'
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
{LL.IMMEDIATE()}
|
{LL.IMMEDIATE()}
|
||||||
</Typography>
|
</Typography>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</ToggleButtonGroup>
|
</ToggleButtonGroup>
|
||||||
|
|
||||||
{scheduleType === ScheduleFlag.SCHEDULE_DAY && (
|
{isDaySchedule && (
|
||||||
<ToggleButtonGroup
|
<ToggleButtonGroup
|
||||||
size="small"
|
size="small"
|
||||||
color="secondary"
|
color="secondary"
|
||||||
value={getFlagDOWstring(editItem.flags)}
|
value={dowFlags}
|
||||||
onChange={(_event, flag: string[]) => {
|
onChange={handleDOWChange}
|
||||||
setEditItem({ ...editItem, flags: getFlagDOWnumber(flag) });
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<ToggleButton value="2">
|
{DAY_FLAGS.map(({ value, flag }) => (
|
||||||
{showDOW(editItem, ScheduleFlag.SCHEDULE_MON)}
|
<ToggleButton key={value} value={value}>
|
||||||
</ToggleButton>
|
{DayOfWeekButton(flag)}
|
||||||
<ToggleButton value="4">
|
|
||||||
{showDOW(editItem, ScheduleFlag.SCHEDULE_TUE)}
|
|
||||||
</ToggleButton>
|
|
||||||
<ToggleButton value="8">
|
|
||||||
{showDOW(editItem, ScheduleFlag.SCHEDULE_WED)}
|
|
||||||
</ToggleButton>
|
|
||||||
<ToggleButton value="16">
|
|
||||||
{showDOW(editItem, ScheduleFlag.SCHEDULE_THU)}
|
|
||||||
</ToggleButton>
|
|
||||||
<ToggleButton value="32">
|
|
||||||
{showDOW(editItem, ScheduleFlag.SCHEDULE_FRI)}
|
|
||||||
</ToggleButton>
|
|
||||||
<ToggleButton value="64">
|
|
||||||
{showDOW(editItem, ScheduleFlag.SCHEDULE_SAT)}
|
|
||||||
</ToggleButton>
|
|
||||||
<ToggleButton value="1">
|
|
||||||
{showDOW(editItem, ScheduleFlag.SCHEDULE_SUN)}
|
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
|
))}
|
||||||
</ToggleButtonGroup>
|
</ToggleButtonGroup>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{scheduleType !== ScheduleFlag.SCHEDULE_IMMEDIATE && (
|
{!isImmediateSchedule && (
|
||||||
<>
|
<>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
<BlockFormControlLabel
|
<BlockFormControlLabel
|
||||||
@@ -284,22 +327,17 @@ const SchedulerDialog = ({
|
|||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
{scheduleType === ScheduleFlag.SCHEDULE_DAY ||
|
{needsTimeField ? (
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_TIMER ? (
|
|
||||||
<>
|
<>
|
||||||
<TextField
|
<TextField
|
||||||
name="time"
|
name="time"
|
||||||
type="time"
|
type="time"
|
||||||
label={
|
label={timeFieldLabel}
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_TIMER
|
value={timeFieldValue}
|
||||||
? LL.TIMER(1)
|
|
||||||
: LL.TIME(1)
|
|
||||||
}
|
|
||||||
value={editItem.time === '' ? '00:00' : editItem.time}
|
|
||||||
margin="normal"
|
margin="normal"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
{scheduleType === ScheduleFlag.SCHEDULE_TIMER && (
|
{isTimerSchedule && (
|
||||||
<Box color="warning.main" ml={2} mt={4}>
|
<Box color="warning.main" ml={2} mt={4}>
|
||||||
<Typography variant="body2">
|
<Typography variant="body2">
|
||||||
{LL.SCHEDULER_HELP_2()}
|
{LL.SCHEDULER_HELP_2()}
|
||||||
@@ -310,16 +348,10 @@ const SchedulerDialog = ({
|
|||||||
) : (
|
) : (
|
||||||
<TextField
|
<TextField
|
||||||
name="time"
|
name="time"
|
||||||
label={
|
label={timeFieldLabel}
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_CONDITION
|
|
||||||
? LL.CONDITION()
|
|
||||||
: scheduleType === ScheduleFlag.SCHEDULE_ONCHANGE
|
|
||||||
? LL.ONCHANGE()
|
|
||||||
: LL.IMMEDIATE()
|
|
||||||
}
|
|
||||||
multiline
|
multiline
|
||||||
fullWidth
|
fullWidth
|
||||||
value={editItem.time === '00:00' ? '' : editItem.time}
|
value={timeFieldValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
@@ -386,7 +418,7 @@ const SchedulerDialog = ({
|
|||||||
>
|
>
|
||||||
{creating ? LL.ADD(0) : LL.UPDATE()}
|
{creating ? LL.ADD(0) : LL.UPDATE()}
|
||||||
</Button>
|
</Button>
|
||||||
{scheduleType === ScheduleFlag.SCHEDULE_IMMEDIATE && editItem.cmd !== '' && (
|
{isImmediateSchedule && editItem.cmd !== '' && (
|
||||||
<Button
|
<Button
|
||||||
startIcon={<PlayArrowIcon />}
|
startIcon={<PlayArrowIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useContext, useState } from 'react';
|
import { useCallback, useContext, useMemo, useRef, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import AddCircleOutlineOutlinedIcon from '@mui/icons-material/AddCircleOutlineOutlined';
|
import AddCircleOutlineOutlinedIcon from '@mui/icons-material/AddCircleOutlineOutlined';
|
||||||
@@ -49,6 +49,74 @@ import {
|
|||||||
temperatureSensorItemValidation
|
temperatureSensorItemValidation
|
||||||
} from './validators';
|
} from './validators';
|
||||||
|
|
||||||
|
// Constants
|
||||||
|
const MS_PER_SECOND = 1000;
|
||||||
|
const MS_PER_MINUTE = 60 * MS_PER_SECOND;
|
||||||
|
const MS_PER_HOUR = 60 * MS_PER_MINUTE;
|
||||||
|
const MS_PER_DAY = 24 * MS_PER_HOUR;
|
||||||
|
const MIN_TEMP_ID = -100;
|
||||||
|
const MAX_TEMP_ID = 100;
|
||||||
|
const GPIO_25 = 25;
|
||||||
|
const GPIO_26 = 26;
|
||||||
|
|
||||||
|
const HEADER_BUTTON_STYLE: React.CSSProperties = {
|
||||||
|
fontSize: '14px',
|
||||||
|
justifyContent: 'flex-start'
|
||||||
|
};
|
||||||
|
|
||||||
|
const HEADER_BUTTON_STYLE_END: React.CSSProperties = {
|
||||||
|
fontSize: '14px',
|
||||||
|
justifyContent: 'flex-end'
|
||||||
|
};
|
||||||
|
|
||||||
|
const common_theme = {
|
||||||
|
BaseRow: `
|
||||||
|
font-size: 14px;
|
||||||
|
.td {
|
||||||
|
height: 32px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
HeaderRow: `
|
||||||
|
text-transform: uppercase;
|
||||||
|
background-color: black;
|
||||||
|
color: #90CAF9;
|
||||||
|
.th {
|
||||||
|
border-bottom: 1px solid #565656;
|
||||||
|
height: 36px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
Row: `
|
||||||
|
background-color: #1e1e1e;
|
||||||
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
|
.td {
|
||||||
|
padding: 8px;
|
||||||
|
border-bottom: 1px solid #565656;
|
||||||
|
}
|
||||||
|
&:hover .td {
|
||||||
|
background-color: #177ac9;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
Cell: `
|
||||||
|
&:last-of-type {
|
||||||
|
text-align: right;
|
||||||
|
},
|
||||||
|
`
|
||||||
|
};
|
||||||
|
|
||||||
|
const temperature_theme_config = {
|
||||||
|
Table: `
|
||||||
|
--data-table-library_grid-template-columns: minmax(0, 1fr) 35%;
|
||||||
|
`
|
||||||
|
};
|
||||||
|
|
||||||
|
const analog_theme_config = {
|
||||||
|
Table: `
|
||||||
|
--data-table-library_grid-template-columns: 80px repeat(1, minmax(0, 1fr)) 120px 110px;
|
||||||
|
`
|
||||||
|
};
|
||||||
|
|
||||||
const Sensors = () => {
|
const Sensors = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
@@ -59,18 +127,22 @@ const Sensors = () => {
|
|||||||
const [temperatureDialogOpen, setTemperatureDialogOpen] = useState<boolean>(false);
|
const [temperatureDialogOpen, setTemperatureDialogOpen] = useState<boolean>(false);
|
||||||
const [analogDialogOpen, setAnalogDialogOpen] = useState<boolean>(false);
|
const [analogDialogOpen, setAnalogDialogOpen] = useState<boolean>(false);
|
||||||
const [creating, setCreating] = useState<boolean>(false);
|
const [creating, setCreating] = useState<boolean>(false);
|
||||||
|
const firstAvailableGPIO = useRef<number>(undefined);
|
||||||
|
|
||||||
const { data: sensorData, send: fetchSensorData } = useRequest(
|
const { data: sensorData, send: fetchSensorData } = useRequest(readSensorData, {
|
||||||
() => readSensorData(),
|
|
||||||
{
|
|
||||||
initialData: {
|
initialData: {
|
||||||
ts: [],
|
ts: [],
|
||||||
as: [],
|
as: [],
|
||||||
analog_enabled: false,
|
analog_enabled: false,
|
||||||
|
available_gpios: [] as number[],
|
||||||
platform: 'ESP32'
|
platform: 'ESP32'
|
||||||
}
|
}
|
||||||
|
}).onSuccess((event) => {
|
||||||
|
// store the first available GPIO in a ref
|
||||||
|
if (event.data.available_gpios.length > 0) {
|
||||||
|
firstAvailableGPIO.current = event.data.available_gpios[0];
|
||||||
}
|
}
|
||||||
);
|
});
|
||||||
|
|
||||||
const { send: sendTemperatureSensor } = useRequest(
|
const { send: sendTemperatureSensor } = useRequest(
|
||||||
(data: WriteTemperatureSensor) => writeTemperatureSensor(data),
|
(data: WriteTemperatureSensor) => writeTemperatureSensor(data),
|
||||||
@@ -86,118 +158,18 @@ const Sensors = () => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
useInterval(() => {
|
const intervalCallback = useCallback(() => {
|
||||||
if (!temperatureDialogOpen && !analogDialogOpen) {
|
if (!temperatureDialogOpen && !analogDialogOpen) {
|
||||||
void fetchSensorData();
|
void fetchSensorData();
|
||||||
}
|
}
|
||||||
});
|
}, [temperatureDialogOpen, analogDialogOpen, fetchSensorData]);
|
||||||
|
|
||||||
const common_theme = useTheme({
|
useInterval(intervalCallback);
|
||||||
BaseRow: `
|
|
||||||
font-size: 14px;
|
|
||||||
.td {
|
|
||||||
height: 32px;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
HeaderRow: `
|
|
||||||
text-transform: uppercase;
|
|
||||||
background-color: black;
|
|
||||||
color: #90CAF9;
|
|
||||||
.th {
|
|
||||||
border-bottom: 1px solid #565656;
|
|
||||||
}
|
|
||||||
.th {
|
|
||||||
height: 36px;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
Row: `
|
|
||||||
background-color: #1e1e1e;
|
|
||||||
position: relative;
|
|
||||||
cursor: pointer;
|
|
||||||
.td {
|
|
||||||
padding: 8px;
|
|
||||||
border-bottom: 1px solid #565656;
|
|
||||||
}
|
|
||||||
&:hover .td {
|
|
||||||
background-color: #177ac9;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
Cell: `
|
|
||||||
&:last-of-type {
|
|
||||||
text-align: right;
|
|
||||||
},
|
|
||||||
`
|
|
||||||
});
|
|
||||||
|
|
||||||
const temperature_theme = useTheme([
|
const temperature_theme = useTheme([common_theme, temperature_theme_config]);
|
||||||
common_theme,
|
const analog_theme = useTheme([common_theme, analog_theme_config]);
|
||||||
{
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: minmax(0, 1fr) 35%;
|
|
||||||
`
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
|
|
||||||
const analog_theme = useTheme([
|
const getSortIcon = useCallback((state: State, sortKey: unknown) => {
|
||||||
common_theme,
|
|
||||||
{
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: 80px repeat(1, minmax(0, 1fr)) 120px 110px;
|
|
||||||
`
|
|
||||||
}
|
|
||||||
]);
|
|
||||||
|
|
||||||
const RenderTemperatureSensors = () => (
|
|
||||||
<Table
|
|
||||||
data={{ nodes: sensorData.ts }}
|
|
||||||
theme={temperature_theme}
|
|
||||||
sort={temperature_sort}
|
|
||||||
layout={{ custom: true }}
|
|
||||||
>
|
|
||||||
{(tableList: TemperatureSensor[]) => (
|
|
||||||
<>
|
|
||||||
<Header>
|
|
||||||
<HeaderRow>
|
|
||||||
<HeaderCell resize>
|
|
||||||
<Button
|
|
||||||
fullWidth
|
|
||||||
style={{ fontSize: '14px', justifyContent: 'flex-start' }}
|
|
||||||
endIcon={getSortIcon(temperature_sort.state, 'NAME')}
|
|
||||||
onClick={() =>
|
|
||||||
temperature_sort.fns.onToggleSort({ sortKey: 'NAME' })
|
|
||||||
}
|
|
||||||
>
|
|
||||||
{LL.NAME(0)}
|
|
||||||
</Button>
|
|
||||||
</HeaderCell>
|
|
||||||
<HeaderCell stiff>
|
|
||||||
<Button
|
|
||||||
fullWidth
|
|
||||||
style={{ fontSize: '14px', justifyContent: 'flex-end' }}
|
|
||||||
endIcon={getSortIcon(temperature_sort.state, 'VALUE')}
|
|
||||||
onClick={() =>
|
|
||||||
temperature_sort.fns.onToggleSort({ sortKey: 'VALUE' })
|
|
||||||
}
|
|
||||||
>
|
|
||||||
{LL.VALUE(0)}
|
|
||||||
</Button>
|
|
||||||
</HeaderCell>
|
|
||||||
</HeaderRow>
|
|
||||||
</Header>
|
|
||||||
<Body>
|
|
||||||
{tableList.map((ts: TemperatureSensor) => (
|
|
||||||
<Row key={ts.id} item={ts} onClick={() => updateTemperatureSensor(ts)}>
|
|
||||||
<Cell>{ts.n}</Cell>
|
|
||||||
<Cell>{formatValue(ts.t, ts.u)}</Cell>
|
|
||||||
</Row>
|
|
||||||
))}
|
|
||||||
</Body>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Table>
|
|
||||||
);
|
|
||||||
|
|
||||||
const getSortIcon = (state: State, sortKey: unknown) => {
|
|
||||||
if (state.sortKey === sortKey && state.reverse) {
|
if (state.sortKey === sortKey && state.reverse) {
|
||||||
return <KeyboardArrowDownOutlinedIcon />;
|
return <KeyboardArrowDownOutlinedIcon />;
|
||||||
}
|
}
|
||||||
@@ -205,7 +177,7 @@ const Sensors = () => {
|
|||||||
return <KeyboardArrowUpOutlinedIcon />;
|
return <KeyboardArrowUpOutlinedIcon />;
|
||||||
}
|
}
|
||||||
return <UnfoldMoreOutlinedIcon />;
|
return <UnfoldMoreOutlinedIcon />;
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const analog_sort = useSort(
|
const analog_sort = useSort(
|
||||||
{ nodes: sensorData.as },
|
{ nodes: sensorData.as },
|
||||||
@@ -218,11 +190,20 @@ const Sensors = () => {
|
|||||||
},
|
},
|
||||||
sortToggleType: SortToggleType.AlternateWithReset,
|
sortToggleType: SortToggleType.AlternateWithReset,
|
||||||
sortFns: {
|
sortFns: {
|
||||||
GPIO: (array) => array.sort((a, b) => a.g - b.g),
|
GPIO: (array) =>
|
||||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-return
|
[...array].sort(
|
||||||
NAME: (array) => array.sort((a, b) => a.n.localeCompare(b.n)),
|
(a, b) => ((a as AnalogSensor)?.g ?? 0) - ((b as AnalogSensor)?.g ?? 0)
|
||||||
TYPE: (array) => array.sort((a, b) => a.t - b.t),
|
),
|
||||||
VALUE: (array) => array.sort((a, b) => a.v - b.v)
|
NAME: (array) =>
|
||||||
|
[...array].sort((a, b) =>
|
||||||
|
((a as AnalogSensor)?.n ?? '').localeCompare(
|
||||||
|
(b as AnalogSensor)?.n ?? ''
|
||||||
|
)
|
||||||
|
),
|
||||||
|
TYPE: (array) =>
|
||||||
|
[...array].sort((a, b) => (a as AnalogSensor).t - (b as AnalogSensor).t),
|
||||||
|
VALUE: (array) =>
|
||||||
|
[...array].sort((a, b) => (a as AnalogSensor).v - (b as AnalogSensor).v)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@@ -238,34 +219,45 @@ const Sensors = () => {
|
|||||||
},
|
},
|
||||||
sortToggleType: SortToggleType.AlternateWithReset,
|
sortToggleType: SortToggleType.AlternateWithReset,
|
||||||
sortFns: {
|
sortFns: {
|
||||||
// eslint-disable-next-line @typescript-eslint/no-unsafe-return
|
NAME: (array) =>
|
||||||
NAME: (array) => array.sort((a, b) => a.n.localeCompare(b.n)),
|
[...array].sort((a, b) =>
|
||||||
VALUE: (array) => array.sort((a, b) => a.t - b.t)
|
(a as TemperatureSensor).n.localeCompare((b as TemperatureSensor).n)
|
||||||
|
),
|
||||||
|
VALUE: (array) =>
|
||||||
|
[...array].sort(
|
||||||
|
(a, b) =>
|
||||||
|
((a as TemperatureSensor).t ?? 0) - ((b as TemperatureSensor).t ?? 0)
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
useLayoutTitle(LL.SENSORS());
|
useLayoutTitle(LL.SENSORS());
|
||||||
|
|
||||||
const formatDurationMin = (duration_min: number) => {
|
const formatDurationMin = useCallback(
|
||||||
const days = Math.trunc((duration_min * 60000) / 86400000);
|
(duration_min: number) => {
|
||||||
const hours = Math.trunc((duration_min * 60000) / 3600000) % 24;
|
const totalMs = duration_min * MS_PER_MINUTE;
|
||||||
const minutes = Math.trunc((duration_min * 60000) / 60000) % 60;
|
const days = Math.trunc(totalMs / MS_PER_DAY);
|
||||||
|
const hours = Math.trunc(totalMs / MS_PER_HOUR) % 24;
|
||||||
|
const minutes = Math.trunc(totalMs / MS_PER_MINUTE) % 60;
|
||||||
|
|
||||||
let formatted = '';
|
const parts: string[] = [];
|
||||||
if (days) {
|
if (days > 0) {
|
||||||
formatted += LL.NUM_DAYS({ num: days }) + ' ';
|
parts.push(LL.NUM_DAYS({ num: days }));
|
||||||
}
|
}
|
||||||
if (hours) {
|
if (hours > 0) {
|
||||||
formatted += LL.NUM_HOURS({ num: hours }) + ' ';
|
parts.push(LL.NUM_HOURS({ num: hours }));
|
||||||
}
|
}
|
||||||
if (minutes) {
|
if (minutes > 0) {
|
||||||
formatted += LL.NUM_MINUTES({ num: minutes });
|
parts.push(LL.NUM_MINUTES({ num: minutes }));
|
||||||
}
|
}
|
||||||
return formatted;
|
return parts.join(' ');
|
||||||
};
|
},
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
function formatValue(value: unknown, uom: DeviceValueUOM) {
|
const formatValue = useCallback(
|
||||||
|
(value: unknown, uom: DeviceValueUOM) => {
|
||||||
if (value === undefined) {
|
if (value === undefined) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
@@ -294,23 +286,34 @@ const Sensors = () => {
|
|||||||
default:
|
default:
|
||||||
return new Intl.NumberFormat().format(value) + ' ' + DeviceValueUOM_s[uom];
|
return new Intl.NumberFormat().format(value) + ' ' + DeviceValueUOM_s[uom];
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
[formatDurationMin, LL]
|
||||||
|
);
|
||||||
|
|
||||||
const updateTemperatureSensor = (ts: TemperatureSensor) => {
|
const updateTemperatureSensor = useCallback(
|
||||||
|
(ts: TemperatureSensor) => {
|
||||||
if (me.admin) {
|
if (me.admin) {
|
||||||
ts.o_n = ts.n;
|
ts.o_n = ts.n;
|
||||||
setSelectedTemperatureSensor(ts);
|
setSelectedTemperatureSensor(ts);
|
||||||
setTemperatureDialogOpen(true);
|
setTemperatureDialogOpen(true);
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[me.admin]
|
||||||
|
);
|
||||||
|
|
||||||
const onTemperatureDialogClose = () => {
|
const onTemperatureDialogClose = useCallback(() => {
|
||||||
setTemperatureDialogOpen(false);
|
setTemperatureDialogOpen(false);
|
||||||
void fetchSensorData();
|
void fetchSensorData();
|
||||||
};
|
}, [fetchSensorData]);
|
||||||
|
|
||||||
const onTemperatureDialogSave = async (ts: TemperatureSensor) => {
|
const onTemperatureDialogSave = useCallback(
|
||||||
await sendTemperatureSensor({ id: ts.id, name: ts.n, offset: ts.o })
|
async (ts: TemperatureSensor) => {
|
||||||
|
await sendTemperatureSensor({
|
||||||
|
id: ts.id,
|
||||||
|
name: ts.n,
|
||||||
|
offset: ts.o,
|
||||||
|
is_system: ts.s
|
||||||
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success(LL.UPDATED_OF(LL.SENSOR(1)));
|
toast.success(LL.UPDATED_OF(LL.SENSOR(1)));
|
||||||
})
|
})
|
||||||
@@ -322,40 +325,51 @@ const Sensors = () => {
|
|||||||
setSelectedTemperatureSensor(undefined);
|
setSelectedTemperatureSensor(undefined);
|
||||||
void fetchSensorData();
|
void fetchSensorData();
|
||||||
});
|
});
|
||||||
};
|
},
|
||||||
|
[sendTemperatureSensor, LL, fetchSensorData]
|
||||||
|
);
|
||||||
|
|
||||||
const updateAnalogSensor = (as: AnalogSensor) => {
|
const updateAnalogSensor = useCallback(
|
||||||
|
(as: AnalogSensor) => {
|
||||||
if (me.admin) {
|
if (me.admin) {
|
||||||
setCreating(false);
|
setCreating(false);
|
||||||
as.o_n = as.n;
|
as.o_n = as.n;
|
||||||
setSelectedAnalogSensor(as);
|
setSelectedAnalogSensor(as);
|
||||||
setAnalogDialogOpen(true);
|
setAnalogDialogOpen(true);
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[me.admin]
|
||||||
|
);
|
||||||
|
|
||||||
const onAnalogDialogClose = () => {
|
const onAnalogDialogClose = useCallback(() => {
|
||||||
setAnalogDialogOpen(false);
|
setAnalogDialogOpen(false);
|
||||||
void fetchSensorData();
|
void fetchSensorData();
|
||||||
};
|
}, [fetchSensorData]);
|
||||||
|
|
||||||
const addAnalogSensor = () => {
|
const addAnalogSensor = useCallback(() => {
|
||||||
|
if (firstAvailableGPIO.current === undefined) {
|
||||||
|
toast.error('No available GPIO found');
|
||||||
|
return;
|
||||||
|
}
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
setSelectedAnalogSensor({
|
setSelectedAnalogSensor({
|
||||||
id: Math.floor(Math.random() * (Math.floor(200) - 100) + 100),
|
id: Math.floor(Math.random() * (MAX_TEMP_ID - MIN_TEMP_ID) + MIN_TEMP_ID),
|
||||||
n: '',
|
n: '',
|
||||||
g: 21, // default GPIO 21 which is safe for all platforms
|
g: firstAvailableGPIO.current,
|
||||||
u: 0,
|
u: DeviceValueUOM.NONE,
|
||||||
v: 0,
|
v: 0,
|
||||||
o: 0,
|
o: 0,
|
||||||
t: 0,
|
|
||||||
f: 1,
|
f: 1,
|
||||||
|
t: AnalogType.DIGITAL_IN, // default to digital in 1
|
||||||
d: false,
|
d: false,
|
||||||
|
s: false,
|
||||||
o_n: ''
|
o_n: ''
|
||||||
});
|
});
|
||||||
setAnalogDialogOpen(true);
|
setAnalogDialogOpen(true);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const onAnalogDialogSave = async (as: AnalogSensor) => {
|
const onAnalogDialogSave = useCallback(
|
||||||
|
async (as: AnalogSensor) => {
|
||||||
await sendAnalogSensor({
|
await sendAnalogSensor({
|
||||||
id: as.id,
|
id: as.id,
|
||||||
gpio: as.g,
|
gpio: as.g,
|
||||||
@@ -364,7 +378,8 @@ const Sensors = () => {
|
|||||||
factor: as.f,
|
factor: as.f,
|
||||||
uom: as.u,
|
uom: as.u,
|
||||||
type: as.t,
|
type: as.t,
|
||||||
deleted: as.d
|
deleted: as.d,
|
||||||
|
is_system: as.s
|
||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success(LL.UPDATED_OF(LL.ANALOG_SENSOR(2)));
|
toast.success(LL.UPDATED_OF(LL.ANALOG_SENSOR(2)));
|
||||||
@@ -377,9 +392,12 @@ const Sensors = () => {
|
|||||||
setSelectedAnalogSensor(undefined);
|
setSelectedAnalogSensor(undefined);
|
||||||
void fetchSensorData();
|
void fetchSensorData();
|
||||||
});
|
});
|
||||||
};
|
},
|
||||||
|
[sendAnalogSensor, LL, fetchSensorData]
|
||||||
|
);
|
||||||
|
|
||||||
const RenderAnalogSensors = () => (
|
const RenderAnalogSensors = useMemo(
|
||||||
|
() => (
|
||||||
<Table
|
<Table
|
||||||
data={{ nodes: sensorData.as }}
|
data={{ nodes: sensorData.as }}
|
||||||
theme={analog_theme}
|
theme={analog_theme}
|
||||||
@@ -393,7 +411,7 @@ const Sensors = () => {
|
|||||||
<HeaderCell stiff>
|
<HeaderCell stiff>
|
||||||
<Button
|
<Button
|
||||||
fullWidth
|
fullWidth
|
||||||
style={{ fontSize: '14px', justifyContent: 'flex-start' }}
|
style={HEADER_BUTTON_STYLE}
|
||||||
endIcon={getSortIcon(analog_sort.state, 'GPIO')}
|
endIcon={getSortIcon(analog_sort.state, 'GPIO')}
|
||||||
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'GPIO' })}
|
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'GPIO' })}
|
||||||
>
|
>
|
||||||
@@ -403,7 +421,7 @@ const Sensors = () => {
|
|||||||
<HeaderCell resize>
|
<HeaderCell resize>
|
||||||
<Button
|
<Button
|
||||||
fullWidth
|
fullWidth
|
||||||
style={{ fontSize: '14px', justifyContent: 'flex-start' }}
|
style={HEADER_BUTTON_STYLE}
|
||||||
endIcon={getSortIcon(analog_sort.state, 'NAME')}
|
endIcon={getSortIcon(analog_sort.state, 'NAME')}
|
||||||
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'NAME' })}
|
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'NAME' })}
|
||||||
>
|
>
|
||||||
@@ -413,7 +431,7 @@ const Sensors = () => {
|
|||||||
<HeaderCell stiff>
|
<HeaderCell stiff>
|
||||||
<Button
|
<Button
|
||||||
fullWidth
|
fullWidth
|
||||||
style={{ fontSize: '14px', justifyContent: 'flex-start' }}
|
style={HEADER_BUTTON_STYLE}
|
||||||
endIcon={getSortIcon(analog_sort.state, 'TYPE')}
|
endIcon={getSortIcon(analog_sort.state, 'TYPE')}
|
||||||
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'TYPE' })}
|
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'TYPE' })}
|
||||||
>
|
>
|
||||||
@@ -423,9 +441,11 @@ const Sensors = () => {
|
|||||||
<HeaderCell stiff>
|
<HeaderCell stiff>
|
||||||
<Button
|
<Button
|
||||||
fullWidth
|
fullWidth
|
||||||
style={{ fontSize: '14px', justifyContent: 'flex-end' }}
|
style={HEADER_BUTTON_STYLE_END}
|
||||||
endIcon={getSortIcon(analog_sort.state, 'VALUE')}
|
endIcon={getSortIcon(analog_sort.state, 'VALUE')}
|
||||||
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'VALUE' })}
|
onClick={() =>
|
||||||
|
analog_sort.fns.onToggleSort({ sortKey: 'VALUE' })
|
||||||
|
}
|
||||||
>
|
>
|
||||||
{LL.VALUE(0)}
|
{LL.VALUE(0)}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -433,17 +453,24 @@ const Sensors = () => {
|
|||||||
</HeaderRow>
|
</HeaderRow>
|
||||||
</Header>
|
</Header>
|
||||||
<Body>
|
<Body>
|
||||||
{tableList.map((a: AnalogSensor) => (
|
{tableList.map((as: AnalogSensor) => (
|
||||||
<Row key={a.id} item={a} onClick={() => updateAnalogSensor(a)}>
|
<Row
|
||||||
<Cell stiff>{a.g}</Cell>
|
style={{ color: as.s ? 'grey' : 'inherit' }}
|
||||||
<Cell>{a.n}</Cell>
|
key={as.id}
|
||||||
<Cell stiff>{AnalogTypeNames[a.t]} </Cell>
|
item={as}
|
||||||
{(a.t === AnalogType.DIGITAL_OUT && a.g !== 25 && a.g !== 26) ||
|
onClick={() => updateAnalogSensor(as)}
|
||||||
a.t === AnalogType.DIGITAL_IN ||
|
>
|
||||||
a.t === AnalogType.PULSE ? (
|
<Cell stiff>{as.g}</Cell>
|
||||||
<Cell stiff>{a.v ? LL.ON() : LL.OFF()}</Cell>
|
<Cell>{as.n}</Cell>
|
||||||
|
<Cell stiff>{AnalogTypeNames[as.t - 1]} </Cell>
|
||||||
|
{(as.t === AnalogType.DIGITAL_OUT &&
|
||||||
|
as.g !== GPIO_25 &&
|
||||||
|
as.g !== GPIO_26) ||
|
||||||
|
as.t === AnalogType.DIGITAL_IN ||
|
||||||
|
as.t === AnalogType.PULSE ? (
|
||||||
|
<Cell stiff>{as.v ? LL.ON() : LL.OFF()}</Cell>
|
||||||
) : (
|
) : (
|
||||||
<Cell stiff>{a.t ? formatValue(a.v, a.u) : ''}</Cell>
|
<Cell stiff>{formatValue(as.v, as.u)}</Cell>
|
||||||
)}
|
)}
|
||||||
</Row>
|
</Row>
|
||||||
))}
|
))}
|
||||||
@@ -451,14 +478,90 @@ const Sensors = () => {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Table>
|
</Table>
|
||||||
|
),
|
||||||
|
[
|
||||||
|
analog_sort,
|
||||||
|
analog_theme,
|
||||||
|
getSortIcon,
|
||||||
|
sensorData.as,
|
||||||
|
LL,
|
||||||
|
updateAnalogSensor,
|
||||||
|
formatValue
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
const RenderTemperatureSensors = useMemo(
|
||||||
|
() => (
|
||||||
|
<Table
|
||||||
|
data={{ nodes: sensorData.ts }}
|
||||||
|
theme={temperature_theme}
|
||||||
|
sort={temperature_sort}
|
||||||
|
layout={{ custom: true }}
|
||||||
|
>
|
||||||
|
{(tableList: TemperatureSensor[]) => (
|
||||||
|
<>
|
||||||
|
<Header>
|
||||||
|
<HeaderRow>
|
||||||
|
<HeaderCell resize>
|
||||||
|
<Button
|
||||||
|
fullWidth
|
||||||
|
style={HEADER_BUTTON_STYLE}
|
||||||
|
endIcon={getSortIcon(temperature_sort.state, 'NAME')}
|
||||||
|
onClick={() =>
|
||||||
|
temperature_sort.fns.onToggleSort({ sortKey: 'NAME' })
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{LL.NAME(0)}
|
||||||
|
</Button>
|
||||||
|
</HeaderCell>
|
||||||
|
<HeaderCell stiff>
|
||||||
|
<Button
|
||||||
|
fullWidth
|
||||||
|
style={HEADER_BUTTON_STYLE_END}
|
||||||
|
endIcon={getSortIcon(temperature_sort.state, 'VALUE')}
|
||||||
|
onClick={() =>
|
||||||
|
temperature_sort.fns.onToggleSort({ sortKey: 'VALUE' })
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{LL.VALUE(0)}
|
||||||
|
</Button>
|
||||||
|
</HeaderCell>
|
||||||
|
</HeaderRow>
|
||||||
|
</Header>
|
||||||
|
<Body>
|
||||||
|
{tableList.map((ts: TemperatureSensor) => (
|
||||||
|
<Row
|
||||||
|
style={{ color: ts.s ? 'grey' : 'inherit' }}
|
||||||
|
key={ts.id}
|
||||||
|
item={ts}
|
||||||
|
onClick={() => updateTemperatureSensor(ts)}
|
||||||
|
>
|
||||||
|
<Cell>{ts.n}</Cell>
|
||||||
|
<Cell>{formatValue(ts.t, ts.u)}</Cell>
|
||||||
|
</Row>
|
||||||
|
))}
|
||||||
|
</Body>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Table>
|
||||||
|
),
|
||||||
|
[
|
||||||
|
temperature_sort,
|
||||||
|
temperature_theme,
|
||||||
|
getSortIcon,
|
||||||
|
sensorData.ts,
|
||||||
|
LL,
|
||||||
|
updateTemperatureSensor,
|
||||||
|
formatValue
|
||||||
|
]
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
<Typography sx={{ pb: 1 }} variant="h6" color="secondary">
|
<Typography sx={{ pb: 1 }} variant="h6" color="primary">
|
||||||
{LL.TEMP_SENSORS()}
|
{LL.TEMP_SENSORS()}
|
||||||
</Typography>
|
</Typography>
|
||||||
<RenderTemperatureSensors />
|
{RenderTemperatureSensors}
|
||||||
{selectedTemperatureSensor && (
|
{selectedTemperatureSensor && (
|
||||||
<DashboardSensorsTemperatureDialog
|
<DashboardSensorsTemperatureDialog
|
||||||
open={temperatureDialogOpen}
|
open={temperatureDialogOpen}
|
||||||
@@ -471,10 +574,10 @@ const Sensors = () => {
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<Typography sx={{ pt: 4, pb: 1 }} variant="h6" color="secondary">
|
<Typography sx={{ pt: 4, pb: 1 }} variant="h6" color="primary">
|
||||||
{LL.ANALOG_SENSORS()}
|
{LL.ANALOG_SENSORS()}
|
||||||
</Typography>
|
</Typography>
|
||||||
<RenderAnalogSensors />
|
{RenderAnalogSensors}
|
||||||
{selectedAnalogSensor && (
|
{selectedAnalogSensor && (
|
||||||
<DashboardSensorsAnalogDialog
|
<DashboardSensorsAnalogDialog
|
||||||
open={analogDialogOpen}
|
open={analogDialogOpen}
|
||||||
@@ -482,16 +585,13 @@ const Sensors = () => {
|
|||||||
onSave={onAnalogDialogSave}
|
onSave={onAnalogDialogSave}
|
||||||
creating={creating}
|
creating={creating}
|
||||||
selectedItem={selectedAnalogSensor}
|
selectedItem={selectedAnalogSensor}
|
||||||
validator={analogSensorItemValidation(
|
analogGPIOList={sensorData.available_gpios}
|
||||||
sensorData.as,
|
disabledTypeList={sensorData.exclude_types}
|
||||||
selectedAnalogSensor,
|
validator={analogSensorItemValidation(sensorData.as, selectedAnalogSensor)}
|
||||||
creating,
|
|
||||||
sensorData.platform
|
|
||||||
)}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{sensorData?.analog_enabled === true && me.admin && (
|
{sensorData?.analog_enabled === true && me.admin && (
|
||||||
<Box mt={1} display="flex" flexWrap="wrap" justifyContent="flex-end">
|
<Box mt={2} display="flex" flexWrap="wrap" justifyContent="flex-end">
|
||||||
<Button
|
<Button
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="primary"
|
color="primary"
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
|
import DoneIcon from '@mui/icons-material/Done';
|
||||||
import RemoveIcon from '@mui/icons-material/RemoveCircleOutline';
|
import RemoveIcon from '@mui/icons-material/RemoveCircleOutline';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
import {
|
import {
|
||||||
@@ -13,7 +14,6 @@ import {
|
|||||||
Grid,
|
Grid,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
TextField,
|
|
||||||
Typography
|
Typography
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
|
|
||||||
@@ -34,6 +34,8 @@ interface DashboardSensorsAnalogDialogProps {
|
|||||||
onSave: (as: AnalogSensor) => void;
|
onSave: (as: AnalogSensor) => void;
|
||||||
creating: boolean;
|
creating: boolean;
|
||||||
selectedItem: AnalogSensor;
|
selectedItem: AnalogSensor;
|
||||||
|
analogGPIOList: number[];
|
||||||
|
disabledTypeList: number[];
|
||||||
validator: Schema;
|
validator: Schema;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -43,13 +45,111 @@ const SensorsAnalogDialog = ({
|
|||||||
onSave,
|
onSave,
|
||||||
creating,
|
creating,
|
||||||
selectedItem,
|
selectedItem,
|
||||||
|
analogGPIOList,
|
||||||
|
disabledTypeList,
|
||||||
validator
|
validator
|
||||||
}: DashboardSensorsAnalogDialogProps) => {
|
}: DashboardSensorsAnalogDialogProps) => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
const [editItem, setEditItem] = useState<AnalogSensor>(selectedItem);
|
const [editItem, setEditItem] = useState<AnalogSensor>(selectedItem);
|
||||||
const updateFormValue = updateValue(setEditItem);
|
|
||||||
|
|
||||||
|
const updateFormValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValue((updater) =>
|
||||||
|
setEditItem(
|
||||||
|
(prev) =>
|
||||||
|
updater(
|
||||||
|
prev as unknown as Record<string, unknown>
|
||||||
|
) as unknown as AnalogSensor
|
||||||
|
)
|
||||||
|
),
|
||||||
|
[setEditItem]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Memoize helper functions to check sensor type conditions
|
||||||
|
const isCounterOrRate = useMemo(
|
||||||
|
() =>
|
||||||
|
editItem.t === AnalogType.COUNTER ||
|
||||||
|
editItem.t === AnalogType.RATE ||
|
||||||
|
(editItem.t >= AnalogType.CNT_0 && editItem.t <= AnalogType.CNT_2),
|
||||||
|
[editItem.t]
|
||||||
|
);
|
||||||
|
const isCounter = useMemo(
|
||||||
|
() =>
|
||||||
|
editItem.t === AnalogType.COUNTER ||
|
||||||
|
(editItem.t >= AnalogType.CNT_0 && editItem.t <= AnalogType.CNT_2),
|
||||||
|
[editItem.t]
|
||||||
|
);
|
||||||
|
const isFreqType = useMemo(
|
||||||
|
() => editItem.t >= AnalogType.FREQ_0 && editItem.t <= AnalogType.FREQ_2,
|
||||||
|
[editItem.t]
|
||||||
|
);
|
||||||
|
const isPWM = useMemo(
|
||||||
|
() =>
|
||||||
|
editItem.t === AnalogType.PWM_0 ||
|
||||||
|
editItem.t === AnalogType.PWM_1 ||
|
||||||
|
editItem.t === AnalogType.PWM_2,
|
||||||
|
[editItem.t]
|
||||||
|
);
|
||||||
|
const isDACOutGPIO = useMemo(
|
||||||
|
() =>
|
||||||
|
editItem.t === AnalogType.DIGITAL_OUT &&
|
||||||
|
(editItem.g === 25 || editItem.g === 26),
|
||||||
|
[editItem.t, editItem.g]
|
||||||
|
);
|
||||||
|
const isDigitalOutGPIO = useMemo(
|
||||||
|
() =>
|
||||||
|
editItem.t === AnalogType.DIGITAL_OUT &&
|
||||||
|
editItem.g !== 25 &&
|
||||||
|
editItem.g !== 26,
|
||||||
|
[editItem.t, editItem.g]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Memoize menu items to avoid recreation on each render
|
||||||
|
const analogTypeMenuItems = useMemo(
|
||||||
|
() =>
|
||||||
|
AnalogTypeNames.map((val, i) => ({ name: val, value: i + 1 }))
|
||||||
|
.sort((a, b) => a.name.localeCompare(b.name))
|
||||||
|
.map(({ name, value }) => (
|
||||||
|
<MenuItem
|
||||||
|
key={name}
|
||||||
|
value={value}
|
||||||
|
disabled={disabledTypeList?.includes(value)}
|
||||||
|
>
|
||||||
|
{name}
|
||||||
|
</MenuItem>
|
||||||
|
)),
|
||||||
|
[disabledTypeList]
|
||||||
|
);
|
||||||
|
|
||||||
|
const uomMenuItems = useMemo(
|
||||||
|
() =>
|
||||||
|
DeviceValueUOM_s.map((val, i) => (
|
||||||
|
<MenuItem key={val} value={i}>
|
||||||
|
{val}
|
||||||
|
</MenuItem>
|
||||||
|
)),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
|
const analogGPIOMenuItems = () =>
|
||||||
|
// add selectedItem.g to the list
|
||||||
|
[
|
||||||
|
...(analogGPIOList?.includes(selectedItem.g) || selectedItem.g === undefined
|
||||||
|
? analogGPIOList
|
||||||
|
: [selectedItem.g, ...analogGPIOList])
|
||||||
|
]
|
||||||
|
.filter((gpio, idx, arr) => arr.indexOf(gpio) === idx)
|
||||||
|
.sort((a, b) => a - b)
|
||||||
|
.map((gpio: number) => {
|
||||||
|
return (
|
||||||
|
<MenuItem key={gpio} value={gpio}>
|
||||||
|
{gpio}
|
||||||
|
</MenuItem>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Reset form when dialog opens or selectedItem changes
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
@@ -57,16 +157,16 @@ const SensorsAnalogDialog = ({
|
|||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleClose = (
|
const handleClose = useCallback(
|
||||||
_event: React.SyntheticEvent,
|
(_event: React.SyntheticEvent, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
reason: 'backdropClick' | 'escapeKeyDown'
|
|
||||||
) => {
|
|
||||||
if (reason !== 'backdropClick') {
|
if (reason !== 'backdropClick') {
|
||||||
onClose();
|
onClose();
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[onClose]
|
||||||
|
);
|
||||||
|
|
||||||
const save = async () => {
|
const save = useCallback(async () => {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, editItem);
|
||||||
@@ -74,97 +174,84 @@ const SensorsAnalogDialog = ({
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
}, [validator, editItem, onSave]);
|
||||||
|
|
||||||
const remove = () => {
|
const remove = useCallback(() => {
|
||||||
editItem.d = true;
|
onSave({ ...editItem, d: true });
|
||||||
onSave(editItem);
|
}, [editItem, onSave]);
|
||||||
};
|
|
||||||
|
const dialogTitle = useMemo(
|
||||||
|
() =>
|
||||||
|
`${creating ? LL.ADD(1) + ' ' + LL.NEW(0) : LL.EDIT()} ${LL.ANALOG_SENSOR(0)}`,
|
||||||
|
[creating, LL]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
<DialogTitle>
|
<DialogTitle>{dialogTitle}</DialogTitle>
|
||||||
{creating ? LL.ADD(1) + ' ' + LL.NEW(0) : LL.EDIT()}
|
|
||||||
{LL.ANALOG_SENSOR(0)}
|
|
||||||
</DialogTitle>
|
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Grid container spacing={2}>
|
<Grid container spacing={2}>
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
name="g"
|
name="g"
|
||||||
label="GPIO"
|
label="GPIO"
|
||||||
sx={{ width: '11ch' }}
|
value={editItem.g}
|
||||||
value={numberValue(editItem.g)}
|
sx={{ width: '9ch' }}
|
||||||
type="number"
|
disabled={editItem.s || !creating}
|
||||||
variant="outlined"
|
select
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
>
|
||||||
</Grid>
|
{analogGPIOMenuItems()}
|
||||||
{creating && (
|
</ValidatedTextField>
|
||||||
<Grid>
|
|
||||||
<Box color="warning.main" mt={2}>
|
|
||||||
<Typography variant="body2">{LL.WARN_GPIO()}</Typography>
|
|
||||||
</Box>
|
|
||||||
</Grid>
|
|
||||||
)}
|
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
name="n"
|
name="n"
|
||||||
label={LL.NAME(0)}
|
label={LL.NAME(0)}
|
||||||
value={editItem.n}
|
value={editItem.n}
|
||||||
fullWidth
|
fullWidth
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="t"
|
name="t"
|
||||||
label={LL.TYPE(0)}
|
label={LL.TYPE(0)}
|
||||||
value={editItem.t}
|
value={editItem.t}
|
||||||
fullWidth
|
fullWidth
|
||||||
select
|
select
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
>
|
>
|
||||||
{AnalogTypeNames.map((val, i) => (
|
{analogTypeMenuItems}
|
||||||
<MenuItem key={val} value={i}>
|
</ValidatedTextField>
|
||||||
{val}
|
|
||||||
</MenuItem>
|
|
||||||
))}
|
|
||||||
</TextField>
|
|
||||||
</Grid>
|
</Grid>
|
||||||
{((editItem.t >= AnalogType.COUNTER && editItem.t <= AnalogType.RATE) ||
|
{(isCounterOrRate ||
|
||||||
(editItem.t >= AnalogType.FREQ_0 &&
|
isFreqType ||
|
||||||
editItem.t <= AnalogType.FREQ_2)) && (
|
editItem.t === AnalogType.ADC ||
|
||||||
|
editItem.t === AnalogType.TIMER) && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="u"
|
name="u"
|
||||||
label={LL.UNIT()}
|
label={LL.UNIT()}
|
||||||
value={editItem.u}
|
value={editItem.u}
|
||||||
sx={{ width: '15ch' }}
|
sx={{ width: '15ch' }}
|
||||||
select
|
select
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
>
|
>
|
||||||
{DeviceValueUOM_s.map((val, i) => (
|
{uomMenuItems}
|
||||||
<MenuItem key={val} value={i}>
|
</ValidatedTextField>
|
||||||
{val}
|
|
||||||
</MenuItem>
|
|
||||||
))}
|
|
||||||
</TextField>
|
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.ADC && (
|
{editItem.t === AnalogType.ADC && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.OFFSET()}
|
label={LL.OFFSET()}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
type="number"
|
type="number"
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
startAdornment: (
|
startAdornment: (
|
||||||
@@ -178,14 +265,14 @@ const SensorsAnalogDialog = ({
|
|||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.NTC && (
|
{editItem.t === AnalogType.NTC && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.OFFSET()}
|
label={LL.OFFSET()}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
type="number"
|
type="number"
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
startAdornment: (
|
startAdornment: (
|
||||||
@@ -197,16 +284,16 @@ const SensorsAnalogDialog = ({
|
|||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.COUNTER && (
|
{isCounter && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.STARTVALUE()}
|
label={LL.STARTVALUE()}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
type="number"
|
type="number"
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
htmlInput: { step: '0.001' }
|
htmlInput: { step: '0.001' }
|
||||||
}}
|
}}
|
||||||
@@ -215,88 +302,90 @@ const SensorsAnalogDialog = ({
|
|||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.RGB && (
|
{editItem.t === AnalogType.RGB && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="o"
|
name="o"
|
||||||
label={'RGB ' + LL.VALUE(0)}
|
label={'RGB ' + LL.VALUE(0)}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
type="number"
|
type="number"
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{editItem.t >= AnalogType.COUNTER && editItem.t <= AnalogType.RATE && (
|
{(isCounterOrRate ||
|
||||||
|
isFreqType ||
|
||||||
|
editItem.t === AnalogType.ADC ||
|
||||||
|
editItem.t === AnalogType.TIMER) && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="f"
|
name="f"
|
||||||
label={LL.FACTOR()}
|
label={LL.FACTOR()}
|
||||||
value={numberValue(editItem.f)}
|
value={numberValue(editItem.f)}
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '14ch' }}
|
||||||
type="number"
|
type="number"
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
htmlInput: { step: '0.001' }
|
htmlInput: { step: '0.001' }
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.DIGITAL_OUT &&
|
{isDACOutGPIO && (
|
||||||
(editItem.g === 25 || editItem.g === 26) && (
|
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.VALUE(0)}
|
label={LL.VALUE(0)}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
type="number"
|
type="number"
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
htmlInput: { min: '0', max: '255', step: '1' }
|
htmlInput: { min: '0', max: '255', step: '1' }
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.DIGITAL_OUT &&
|
{isDigitalOutGPIO && (
|
||||||
editItem.g !== 25 &&
|
|
||||||
editItem.g !== 26 && (
|
|
||||||
<>
|
<>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.VALUE(0)}
|
label={LL.VALUE(0)}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
select
|
select
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
>
|
>
|
||||||
<MenuItem value={0}>{LL.OFF()}</MenuItem>
|
<MenuItem value={0}>{LL.OFF()}</MenuItem>
|
||||||
<MenuItem value={1}>{LL.ON()}</MenuItem>
|
<MenuItem value={1}>{LL.ON()}</MenuItem>
|
||||||
</TextField>
|
</ValidatedTextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="f"
|
name="f"
|
||||||
label={LL.POLARITY()}
|
label={LL.POLARITY()}
|
||||||
value={editItem.f}
|
value={editItem.f}
|
||||||
sx={{ width: '15ch' }}
|
sx={{ width: '15ch' }}
|
||||||
select
|
select
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
>
|
>
|
||||||
<MenuItem value={1}>{LL.ACTIVEHIGH()}</MenuItem>
|
<MenuItem value={1}>{LL.ACTIVEHIGH()}</MenuItem>
|
||||||
<MenuItem value={-1}>{LL.ACTIVELOW()}</MenuItem>
|
<MenuItem value={-1}>{LL.ACTIVELOW()}</MenuItem>
|
||||||
</TextField>
|
</ValidatedTextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="u"
|
name="u"
|
||||||
label={LL.STARTVALUE()}
|
label={LL.STARTVALUE()}
|
||||||
sx={{ width: '15ch' }}
|
sx={{ width: '15ch' }}
|
||||||
value={editItem.u}
|
value={editItem.u}
|
||||||
select
|
select
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
>
|
>
|
||||||
<MenuItem value={0}>{LL.UNCHANGED()}</MenuItem>
|
<MenuItem value={0}>{LL.UNCHANGED()}</MenuItem>
|
||||||
<MenuItem value={1}>
|
<MenuItem value={1}>
|
||||||
@@ -305,23 +394,21 @@ const SensorsAnalogDialog = ({
|
|||||||
<MenuItem value={2}>
|
<MenuItem value={2}>
|
||||||
{LL.ALWAYS()} {LL.ON()}
|
{LL.ALWAYS()} {LL.ON()}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
</TextField>
|
</ValidatedTextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{(editItem.t === AnalogType.PWM_0 ||
|
{isPWM && (
|
||||||
editItem.t === AnalogType.PWM_1 ||
|
|
||||||
editItem.t === AnalogType.PWM_2) && (
|
|
||||||
<>
|
<>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="f"
|
name="f"
|
||||||
label={LL.FREQ()}
|
label={LL.FREQ()}
|
||||||
value={numberValue(editItem.f)}
|
value={numberValue(editItem.f)}
|
||||||
type="number"
|
type="number"
|
||||||
variant="outlined"
|
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
startAdornment: (
|
startAdornment: (
|
||||||
@@ -333,14 +420,14 @@ const SensorsAnalogDialog = ({
|
|||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.DUTY_CYCLE()}
|
label={LL.DUTY_CYCLE()}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
type="number"
|
type="number"
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
startAdornment: (
|
startAdornment: (
|
||||||
@@ -356,27 +443,28 @@ const SensorsAnalogDialog = ({
|
|||||||
{editItem.t === AnalogType.PULSE && (
|
{editItem.t === AnalogType.PULSE && (
|
||||||
<>
|
<>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.POLARITY()}
|
label={LL.POLARITY()}
|
||||||
value={editItem.o}
|
value={editItem.o}
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
select
|
select
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
>
|
>
|
||||||
<MenuItem value={0}>{LL.ACTIVEHIGH()}</MenuItem>
|
<MenuItem value={0}>{LL.ACTIVEHIGH()}</MenuItem>
|
||||||
<MenuItem value={1}>{LL.ACTIVELOW()}</MenuItem>
|
<MenuItem value={1}>{LL.ACTIVELOW()}</MenuItem>
|
||||||
</TextField>
|
</ValidatedTextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<ValidatedTextField
|
||||||
name="f"
|
name="f"
|
||||||
label="Pulse"
|
label="Pulse"
|
||||||
value={numberValue(editItem.f)}
|
value={numberValue(editItem.f)}
|
||||||
type="number"
|
type="number"
|
||||||
sx={{ width: '15ch' }}
|
sx={{ width: '15ch' }}
|
||||||
variant="outlined"
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
|
disabled={editItem.s}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
startAdornment: (
|
startAdornment: (
|
||||||
@@ -390,12 +478,43 @@ const SensorsAnalogDialog = ({
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
|
{fieldErrors && Object.keys(fieldErrors).length > 0 && (
|
||||||
|
<Box mt={1}>
|
||||||
|
{Object.values(fieldErrors).map((errArr, idx) =>
|
||||||
|
Array.isArray(errArr)
|
||||||
|
? errArr.map((err, j) => (
|
||||||
|
<Typography
|
||||||
|
key={`${idx}-${j}`}
|
||||||
|
color="error"
|
||||||
|
variant="caption"
|
||||||
|
display="block"
|
||||||
|
>
|
||||||
|
{err.message}
|
||||||
|
</Typography>
|
||||||
|
))
|
||||||
|
: null
|
||||||
|
)}
|
||||||
|
</Box>
|
||||||
|
)}
|
||||||
|
{editItem.s && (
|
||||||
|
<Grid>
|
||||||
|
<Typography mt={1} color="warning.main" variant="body2">
|
||||||
|
<WarningIcon
|
||||||
|
fontSize="small"
|
||||||
|
sx={{ mr: 1, verticalAlign: 'middle' }}
|
||||||
|
color="warning"
|
||||||
|
/>
|
||||||
|
{LL.SYSTEM(0)} {LL.SENSOR(0)}
|
||||||
|
</Typography>
|
||||||
|
</Grid>
|
||||||
|
)}
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
{!creating && (
|
{!creating && (
|
||||||
<Box flexGrow={1} sx={{ '& button': { mt: 0 } }}>
|
<Box flexGrow={1} sx={{ '& button': { mt: 0 } }}>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<RemoveIcon />}
|
startIcon={<RemoveIcon />}
|
||||||
|
disabled={editItem.s}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="warning"
|
color="warning"
|
||||||
onClick={remove}
|
onClick={remove}
|
||||||
@@ -413,7 +532,7 @@ const SensorsAnalogDialog = ({
|
|||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<WarningIcon color="warning" />}
|
startIcon={<DoneIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={save}
|
onClick={save}
|
||||||
color="primary"
|
color="primary"
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
|
import DoneIcon from '@mui/icons-material/Done';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
import {
|
import {
|
||||||
Box,
|
Box,
|
||||||
@@ -33,6 +34,12 @@ interface SensorsTemperatureDialogProps {
|
|||||||
validator: Schema;
|
validator: Schema;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Constants
|
||||||
|
const OFFSET_MIN = -5;
|
||||||
|
const OFFSET_MAX = 5;
|
||||||
|
const OFFSET_STEP = 0.1;
|
||||||
|
const TEMP_UNIT = '°C';
|
||||||
|
|
||||||
const SensorsTemperatureDialog = ({
|
const SensorsTemperatureDialog = ({
|
||||||
open,
|
open,
|
||||||
onClose,
|
onClose,
|
||||||
@@ -43,7 +50,18 @@ const SensorsTemperatureDialog = ({
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
const [editItem, setEditItem] = useState<TemperatureSensor>(selectedItem);
|
const [editItem, setEditItem] = useState<TemperatureSensor>(selectedItem);
|
||||||
const updateFormValue = updateValue(setEditItem);
|
|
||||||
|
const updateFormValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValue(
|
||||||
|
setEditItem as unknown as (
|
||||||
|
updater: (
|
||||||
|
prevState: Readonly<Record<string, unknown>>
|
||||||
|
) => Record<string, unknown>
|
||||||
|
) => void
|
||||||
|
),
|
||||||
|
[setEditItem]
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
@@ -52,16 +70,16 @@ const SensorsTemperatureDialog = ({
|
|||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleClose = (
|
const handleClose = useCallback(
|
||||||
_event: React.SyntheticEvent,
|
(_event: React.SyntheticEvent, reason?: string) => {
|
||||||
reason: 'backdropClick' | 'escapeKeyDown'
|
|
||||||
) => {
|
|
||||||
if (reason !== 'backdropClick') {
|
if (reason !== 'backdropClick') {
|
||||||
onClose();
|
onClose();
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[onClose]
|
||||||
|
);
|
||||||
|
|
||||||
const save = async () => {
|
const save = useCallback(async () => {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, editItem);
|
||||||
@@ -69,15 +87,31 @@ const SensorsTemperatureDialog = ({
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
}, [validator, editItem, onSave]);
|
||||||
|
|
||||||
|
const dialogTitle = useMemo(() => `${LL.EDIT()} ${LL.TEMP_SENSOR()}`, [LL]);
|
||||||
|
|
||||||
|
const offsetValue = useMemo(() => numberValue(editItem.o), [editItem.o]);
|
||||||
|
|
||||||
|
const slotProps = useMemo(
|
||||||
|
() => ({
|
||||||
|
input: {
|
||||||
|
startAdornment: <InputAdornment position="start">{TEMP_UNIT}</InputAdornment>
|
||||||
|
},
|
||||||
|
htmlInput: {
|
||||||
|
min: OFFSET_MIN,
|
||||||
|
max: OFFSET_MAX,
|
||||||
|
step: OFFSET_STEP
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
<DialogTitle>
|
<DialogTitle>{dialogTitle}</DialogTitle>
|
||||||
{LL.EDIT()} {LL.TEMP_SENSOR()}
|
|
||||||
</DialogTitle>
|
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Box color="warning.main" p={0} pl={0} pr={0} mt={0} mb={2}>
|
<Box color="warning.main" mb={2}>
|
||||||
<Typography variant="body2">
|
<Typography variant="body2">
|
||||||
{LL.ID_OF(LL.SENSOR(0))}: {editItem.id}
|
{LL.ID_OF(LL.SENSOR(0))}: {editItem.id}
|
||||||
</Typography>
|
</Typography>
|
||||||
@@ -85,7 +119,7 @@ const SensorsTemperatureDialog = ({
|
|||||||
<Grid container spacing={2}>
|
<Grid container spacing={2}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors ?? {}}
|
||||||
name="n"
|
name="n"
|
||||||
label={LL.NAME(0)}
|
label={LL.NAME(0)}
|
||||||
value={editItem.n}
|
value={editItem.n}
|
||||||
@@ -97,22 +131,27 @@ const SensorsTemperatureDialog = ({
|
|||||||
<TextField
|
<TextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.OFFSET()}
|
label={LL.OFFSET()}
|
||||||
value={numberValue(editItem.o)}
|
value={offsetValue}
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
type="number"
|
type="number"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
slotProps={{
|
slotProps={slotProps}
|
||||||
input: {
|
|
||||||
startAdornment: (
|
|
||||||
<InputAdornment position="start">°C</InputAdornment>
|
|
||||||
)
|
|
||||||
},
|
|
||||||
htmlInput: { min: '-5', max: '5', step: '0.1' }
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
{editItem.s && (
|
||||||
|
<Grid>
|
||||||
|
<Typography mt={1} color="warning.main" variant="body2">
|
||||||
|
<WarningIcon
|
||||||
|
fontSize="small"
|
||||||
|
sx={{ mr: 1, verticalAlign: 'middle' }}
|
||||||
|
color="warning"
|
||||||
|
/>
|
||||||
|
{LL.SYSTEM(0)} {LL.SENSOR(0)}
|
||||||
|
</Typography>
|
||||||
|
</Grid>
|
||||||
|
)}
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
<Button
|
<Button
|
||||||
@@ -124,7 +163,7 @@ const SensorsTemperatureDialog = ({
|
|||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<WarningIcon color="warning" />}
|
startIcon={<DoneIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={save}
|
onClick={save}
|
||||||
color="primary"
|
color="primary"
|
||||||
|
|||||||
65
interface/src/app/main/UserProfile.tsx
Normal file
65
interface/src/app/main/UserProfile.tsx
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
import { memo, useCallback, useContext } from 'react';
|
||||||
|
|
||||||
|
import PersonIcon from '@mui/icons-material/Person';
|
||||||
|
import {
|
||||||
|
Avatar,
|
||||||
|
Box,
|
||||||
|
Button,
|
||||||
|
Divider,
|
||||||
|
List,
|
||||||
|
ListItem,
|
||||||
|
ListItemText,
|
||||||
|
Typography
|
||||||
|
} from '@mui/material';
|
||||||
|
|
||||||
|
import { AuthenticatedContext } from '@/contexts/authentication';
|
||||||
|
import { SectionContent, useLayoutTitle } from 'components';
|
||||||
|
import { LanguageSelector } from 'components/inputs';
|
||||||
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
|
|
||||||
|
const UserProfileComponent = () => {
|
||||||
|
const { LL } = useI18nContext();
|
||||||
|
const { me, signOut } = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
|
useLayoutTitle(LL.USER_PROFILE());
|
||||||
|
|
||||||
|
const handleSignOut = useCallback(() => {
|
||||||
|
signOut(true);
|
||||||
|
}, [signOut]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SectionContent>
|
||||||
|
<List sx={{ flexGrow: 1 }}>
|
||||||
|
<ListItem disablePadding>
|
||||||
|
<Avatar sx={{ bgcolor: '#9e9e9e', color: 'white' }}>
|
||||||
|
<PersonIcon />
|
||||||
|
</Avatar>
|
||||||
|
<ListItemText
|
||||||
|
sx={{ pl: 2, color: '#2196f3' }}
|
||||||
|
primary={me.username}
|
||||||
|
secondary={'(' + (me.admin ? LL.ADMINISTRATOR() : LL.GUEST()) + ')'}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
</List>
|
||||||
|
<Box mt={2} mb={2} display="flex" alignItems="center">
|
||||||
|
<Typography mr={2} variant="body1" align="center">
|
||||||
|
{LL.LANGUAGE()}:
|
||||||
|
</Typography>
|
||||||
|
<LanguageSelector />
|
||||||
|
</Box>
|
||||||
|
<Divider />
|
||||||
|
<Button
|
||||||
|
sx={{ mt: 2 }}
|
||||||
|
variant="outlined"
|
||||||
|
color="primary"
|
||||||
|
onClick={handleSignOut}
|
||||||
|
>
|
||||||
|
{LL.SIGN_OUT()}
|
||||||
|
</Button>
|
||||||
|
</SectionContent>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const UserProfile = memo(UserProfileComponent);
|
||||||
|
|
||||||
|
export default UserProfile;
|
||||||
@@ -2,27 +2,30 @@ import type { TranslationFunctions } from 'i18n/i18n-types';
|
|||||||
|
|
||||||
import { DeviceValueUOM, DeviceValueUOM_s } from './types';
|
import { DeviceValueUOM, DeviceValueUOM_s } from './types';
|
||||||
|
|
||||||
|
// Cache NumberFormat instances for better performance
|
||||||
|
const numberFormatter = new Intl.NumberFormat();
|
||||||
|
const numberFormatterWithDecimal = new Intl.NumberFormat(undefined, {
|
||||||
|
minimumFractionDigits: 1
|
||||||
|
});
|
||||||
|
|
||||||
const formatDurationMin = (LL: TranslationFunctions, duration_min: number) => {
|
const formatDurationMin = (LL: TranslationFunctions, duration_min: number) => {
|
||||||
const days = Math.trunc((duration_min * 60000) / 86400000);
|
const totalMs = duration_min * 60000;
|
||||||
const hours = Math.trunc((duration_min * 60000) / 3600000) % 24;
|
const days = Math.trunc(totalMs / 86400000);
|
||||||
const minutes = Math.trunc((duration_min * 60000) / 60000) % 60;
|
const hours = Math.trunc(totalMs / 3600000) % 24;
|
||||||
|
const minutes = Math.trunc(duration_min) % 60;
|
||||||
|
|
||||||
let formatted = '';
|
const parts: string[] = [];
|
||||||
if (days) {
|
if (days) {
|
||||||
formatted += LL.NUM_DAYS({ num: days });
|
parts.push(LL.NUM_DAYS({ num: days }));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hours) {
|
if (hours) {
|
||||||
if (formatted) formatted += ' ';
|
parts.push(LL.NUM_HOURS({ num: hours }));
|
||||||
formatted += LL.NUM_HOURS({ num: hours });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (minutes) {
|
if (minutes) {
|
||||||
if (formatted) formatted += ' ';
|
parts.push(LL.NUM_MINUTES({ num: minutes }));
|
||||||
formatted += LL.NUM_MINUTES({ num: minutes });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return formatted;
|
return parts.join(' ');
|
||||||
};
|
};
|
||||||
|
|
||||||
export function formatValue(
|
export function formatValue(
|
||||||
@@ -30,18 +33,21 @@ export function formatValue(
|
|||||||
value?: unknown,
|
value?: unknown,
|
||||||
uom?: DeviceValueUOM
|
uom?: DeviceValueUOM
|
||||||
) {
|
) {
|
||||||
|
// Handle non-numeric values or missing data
|
||||||
if (typeof value !== 'number' || uom === undefined || value === undefined) {
|
if (typeof value !== 'number' || uom === undefined || value === undefined) {
|
||||||
if (value === undefined || typeof value === 'boolean') {
|
if (value === undefined || typeof value === 'boolean') {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
// Type assertion is safe here since we know it's not a number, boolean, or undefined
|
||||||
return (
|
return (
|
||||||
(value as string) +
|
(value as string) +
|
||||||
(value === '' || uom === undefined || uom === 0
|
(value === '' || uom === undefined || uom === DeviceValueUOM.NONE
|
||||||
? ''
|
? ''
|
||||||
: ' ' + DeviceValueUOM_s[uom])
|
: ' ' + DeviceValueUOM_s[uom])
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Handle numeric values
|
||||||
switch (uom) {
|
switch (uom) {
|
||||||
case DeviceValueUOM.HOURS:
|
case DeviceValueUOM.HOURS:
|
||||||
return value ? formatDurationMin(LL, value * 60) : LL.NUM_HOURS({ num: 0 });
|
return value ? formatDurationMin(LL, value * 60) : LL.NUM_HOURS({ num: 0 });
|
||||||
@@ -50,18 +56,12 @@ export function formatValue(
|
|||||||
case DeviceValueUOM.SECONDS:
|
case DeviceValueUOM.SECONDS:
|
||||||
return LL.NUM_SECONDS({ num: value });
|
return LL.NUM_SECONDS({ num: value });
|
||||||
case DeviceValueUOM.NONE:
|
case DeviceValueUOM.NONE:
|
||||||
return new Intl.NumberFormat().format(value);
|
return numberFormatter.format(value);
|
||||||
case DeviceValueUOM.DEGREES:
|
case DeviceValueUOM.DEGREES:
|
||||||
case DeviceValueUOM.DEGREES_R:
|
case DeviceValueUOM.DEGREES_R:
|
||||||
case DeviceValueUOM.FAHRENHEIT:
|
case DeviceValueUOM.FAHRENHEIT:
|
||||||
return (
|
return numberFormatterWithDecimal.format(value) + ' ' + DeviceValueUOM_s[uom];
|
||||||
new Intl.NumberFormat(undefined, {
|
|
||||||
minimumFractionDigits: 1
|
|
||||||
}).format(value) +
|
|
||||||
' ' +
|
|
||||||
DeviceValueUOM_s[uom]
|
|
||||||
);
|
|
||||||
default:
|
default:
|
||||||
return new Intl.NumberFormat().format(value) + ' ' + DeviceValueUOM_s[uom];
|
return numberFormatter.format(value) + ' ' + DeviceValueUOM_s[uom];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -43,6 +43,16 @@ export interface Settings {
|
|||||||
modbus_port: number;
|
modbus_port: number;
|
||||||
modbus_max_clients: number;
|
modbus_max_clients: number;
|
||||||
modbus_timeout: number;
|
modbus_timeout: number;
|
||||||
|
email_enabled: boolean;
|
||||||
|
email_ssl?: boolean;
|
||||||
|
email_starttls?: boolean;
|
||||||
|
email_server: string;
|
||||||
|
email_port: number;
|
||||||
|
email_login: string;
|
||||||
|
email_pass: string;
|
||||||
|
email_sender: string;
|
||||||
|
email_recp: string;
|
||||||
|
email_subject: string;
|
||||||
developer_mode: boolean;
|
developer_mode: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -60,7 +70,7 @@ export interface Stat {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Activity {
|
export interface Activity {
|
||||||
stats: Stat[];
|
readonly stats: readonly Stat[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Device {
|
export interface Device {
|
||||||
@@ -82,38 +92,43 @@ export interface TemperatureSensor {
|
|||||||
t?: number; // temp, optional
|
t?: number; // temp, optional
|
||||||
o: number; // offset
|
o: number; // offset
|
||||||
u: number; // uom
|
u: number; // uom
|
||||||
|
s: boolean; // system sensor flag
|
||||||
o_n?: string;
|
o_n?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AnalogSensor {
|
export interface AnalogSensor {
|
||||||
id: number;
|
id: number;
|
||||||
g: number; // GPIO
|
g: number; // GPIO
|
||||||
n: string;
|
n: string; // name
|
||||||
v: number;
|
v: number; // value
|
||||||
u: number;
|
u: number; // uom
|
||||||
o: number;
|
o: number; // offset
|
||||||
f: number;
|
f: number; // factor
|
||||||
t: number;
|
t: number; // type
|
||||||
d: boolean; // deleted flag
|
d: boolean; // deleted flag
|
||||||
o_n?: string;
|
s: boolean; // system sensor flag
|
||||||
|
o_n?: string; // original name
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface WriteTemperatureSensor {
|
export interface WriteTemperatureSensor {
|
||||||
id: string;
|
id: string;
|
||||||
name: string;
|
name: string;
|
||||||
offset: number;
|
offset: number;
|
||||||
|
is_system: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SensorData {
|
export interface SensorData {
|
||||||
ts: TemperatureSensor[];
|
ts: TemperatureSensor[];
|
||||||
as: AnalogSensor[];
|
as: AnalogSensor[];
|
||||||
analog_enabled: boolean;
|
analog_enabled: boolean;
|
||||||
|
available_gpios: number[];
|
||||||
|
exclude_types: number[];
|
||||||
platform: string;
|
platform: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface CoreData {
|
export interface CoreData {
|
||||||
connected: boolean;
|
readonly connected: boolean;
|
||||||
devices: Device[];
|
readonly devices: readonly Device[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DashboardItem {
|
export interface DashboardItem {
|
||||||
@@ -122,11 +137,12 @@ export interface DashboardItem {
|
|||||||
n?: string; // name, optional
|
n?: string; // name, optional
|
||||||
dv?: DeviceValue; // device value, optional
|
dv?: DeviceValue; // device value, optional
|
||||||
nodes?: DashboardItem[]; // children nodes, optional
|
nodes?: DashboardItem[]; // children nodes, optional
|
||||||
|
parentNode: DashboardItem; // to stop lint errors
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DashboardData {
|
export interface DashboardData {
|
||||||
connected: boolean; // true if connected to EMS bus
|
readonly connected: boolean; // true if connected to EMS bus
|
||||||
nodes: DashboardItem[];
|
readonly nodes: readonly DashboardItem[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DeviceValue {
|
export interface DeviceValue {
|
||||||
@@ -139,10 +155,11 @@ export interface DeviceValue {
|
|||||||
s?: string; // steps for up/down, optional
|
s?: string; // steps for up/down, optional
|
||||||
m?: number; // min, optional
|
m?: number; // min, optional
|
||||||
x?: number; // max, optional
|
x?: number; // max, optional
|
||||||
|
[key: string]: unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DeviceData {
|
export interface DeviceData {
|
||||||
nodes: DeviceValue[];
|
readonly nodes: readonly DeviceValue[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DeviceEntity {
|
export interface DeviceEntity {
|
||||||
@@ -189,13 +206,13 @@ export enum DeviceValueUOM {
|
|||||||
MBAR,
|
MBAR,
|
||||||
LH,
|
LH,
|
||||||
CTKWH,
|
CTKWH,
|
||||||
HZ
|
HERTZ
|
||||||
}
|
}
|
||||||
|
|
||||||
export const DeviceValueUOM_s = [
|
export const DeviceValueUOM_s = [
|
||||||
'',
|
'',
|
||||||
'°C',
|
'°C',
|
||||||
'°C',
|
'°C Rel',
|
||||||
'%',
|
'%',
|
||||||
'l/min',
|
'l/min',
|
||||||
'kWh',
|
'kWh',
|
||||||
@@ -221,11 +238,10 @@ export const DeviceValueUOM_s = [
|
|||||||
'l/h',
|
'l/h',
|
||||||
'ct/kWh',
|
'ct/kWh',
|
||||||
'Hz'
|
'Hz'
|
||||||
];
|
] as const;
|
||||||
|
|
||||||
export enum AnalogType {
|
export enum AnalogType {
|
||||||
REMOVED = -1,
|
REMOVED = -1,
|
||||||
NOTUSED = 0,
|
|
||||||
DIGITAL_IN = 1,
|
DIGITAL_IN = 1,
|
||||||
COUNTER = 2,
|
COUNTER = 2,
|
||||||
ADC = 3,
|
ADC = 3,
|
||||||
@@ -240,31 +256,34 @@ export enum AnalogType {
|
|||||||
PULSE = 12,
|
PULSE = 12,
|
||||||
FREQ_0 = 13,
|
FREQ_0 = 13,
|
||||||
FREQ_1 = 14,
|
FREQ_1 = 14,
|
||||||
FREQ_2 = 15
|
FREQ_2 = 15,
|
||||||
|
CNT_0 = 16,
|
||||||
|
CNT_1 = 17,
|
||||||
|
CNT_2 = 18
|
||||||
}
|
}
|
||||||
|
|
||||||
export const AnalogTypeNames = [
|
export const AnalogTypeNames = [
|
||||||
'(disabled)',
|
'Digital In', // 1
|
||||||
'Digital In',
|
'Counter', // 2
|
||||||
'Counter',
|
'ADC In', // 3
|
||||||
'ADC In',
|
'Timer', // 4
|
||||||
'Timer',
|
'Rate', // 5
|
||||||
'Rate',
|
'Digital Out', // 6
|
||||||
'Digital Out',
|
'PWM 0', // 7
|
||||||
'PWM 0',
|
'PWM 1', // 8
|
||||||
'PWM 1',
|
'PWM 2', // 9
|
||||||
'PWM 2',
|
'NTC Temp', // 10
|
||||||
'NTC Temp.',
|
'RGB Led', // 11
|
||||||
'RGB Led',
|
'Pulse', // 12
|
||||||
'Pulse',
|
'Freq 0', // 13
|
||||||
'Freq 0',
|
'Freq 1', // 14
|
||||||
'Freq 1',
|
'Freq 2', // 15
|
||||||
'Freq 2'
|
'Counter 0', // 16
|
||||||
];
|
'Counter 1', // 17
|
||||||
|
'Counter 2' // 18
|
||||||
|
] as const;
|
||||||
|
|
||||||
type BoardProfiles = Record<string, string>;
|
export const BOARD_PROFILES = {
|
||||||
|
|
||||||
export const BOARD_PROFILES: BoardProfiles = {
|
|
||||||
S32: 'BBQKees Gateway S32',
|
S32: 'BBQKees Gateway S32',
|
||||||
S32S3: 'BBQKees Gateway S3',
|
S32S3: 'BBQKees Gateway S3',
|
||||||
E32: 'BBQKees Gateway E32',
|
E32: 'BBQKees Gateway E32',
|
||||||
@@ -278,7 +297,9 @@ export const BOARD_PROFILES: BoardProfiles = {
|
|||||||
C3MINI: 'Wemos C3 Mini',
|
C3MINI: 'Wemos C3 Mini',
|
||||||
S2MINI: 'Wemos S2 Mini',
|
S2MINI: 'Wemos S2 Mini',
|
||||||
S3MINI: 'Liligo S3'
|
S3MINI: 'Liligo S3'
|
||||||
};
|
} as const;
|
||||||
|
|
||||||
|
export type BoardProfileKey = keyof typeof BOARD_PROFILES;
|
||||||
|
|
||||||
export interface BoardProfile {
|
export interface BoardProfile {
|
||||||
board_profile: string;
|
board_profile: string;
|
||||||
@@ -315,6 +336,7 @@ export interface WriteAnalogSensor {
|
|||||||
uom: number;
|
uom: number;
|
||||||
type: number;
|
type: number;
|
||||||
deleted: boolean;
|
deleted: boolean;
|
||||||
|
is_system: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum DeviceEntityMask {
|
export enum DeviceEntityMask {
|
||||||
@@ -346,7 +368,7 @@ export interface ScheduleItem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Schedule {
|
export interface Schedule {
|
||||||
schedule: ScheduleItem[];
|
readonly schedule: readonly ScheduleItem[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ModuleItem {
|
export interface ModuleItem {
|
||||||
@@ -364,7 +386,7 @@ export interface ModuleItem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Modules {
|
export interface Modules {
|
||||||
modules: ModuleItem[];
|
readonly modules: readonly ModuleItem[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum ScheduleFlag {
|
export enum ScheduleFlag {
|
||||||
@@ -413,7 +435,7 @@ export interface EntityItem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Entities {
|
export interface Entities {
|
||||||
entities: EntityItem[];
|
readonly entities: readonly EntityItem[];
|
||||||
}
|
}
|
||||||
|
|
||||||
// matches emsdevice.h DeviceType
|
// matches emsdevice.h DeviceType
|
||||||
@@ -469,4 +491,4 @@ export const DeviceValueTypeNames = [
|
|||||||
'ENUM',
|
'ENUM',
|
||||||
'RAW',
|
'RAW',
|
||||||
'CMD'
|
'CMD'
|
||||||
];
|
] as const;
|
||||||
|
|||||||
@@ -11,273 +11,158 @@ import type {
|
|||||||
TemperatureSensor
|
TemperatureSensor
|
||||||
} from './types';
|
} from './types';
|
||||||
|
|
||||||
export const GPIO_VALIDATOR = {
|
// Constants
|
||||||
validator(
|
const ERROR_MESSAGES = {
|
||||||
_rule: InternalRuleItem,
|
GPIO_INVALID: 'Must be an valid GPIO port',
|
||||||
value: number,
|
NAME_DUPLICATE: 'Name already in use',
|
||||||
callback: (error?: string) => void
|
GPIO_DUPLICATE: 'GPIO already in use',
|
||||||
) {
|
VALUE_OUT_OF_RANGE: 'Value out of range',
|
||||||
if (
|
HEX_REQUIRED: 'Is required and must be in hex format'
|
||||||
value &&
|
} as const;
|
||||||
(value === 1 ||
|
|
||||||
(value >= 6 && value <= 11) ||
|
|
||||||
value === 20 ||
|
|
||||||
value === 24 ||
|
|
||||||
(value >= 28 && value <= 31) ||
|
|
||||||
value > 40 ||
|
|
||||||
value < 0)
|
|
||||||
) {
|
|
||||||
callback('Must be an valid GPIO port');
|
|
||||||
} else {
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const GPIO_VALIDATORR = {
|
const VALIDATION_LIMITS = {
|
||||||
validator(
|
PORT_MIN: 0,
|
||||||
_rule: InternalRuleItem,
|
PORT_MAX: 65535,
|
||||||
value: number,
|
MODBUS_MAX_CLIENTS_MIN: 0,
|
||||||
callback: (error?: string) => void
|
MODBUS_MAX_CLIENTS_MAX: 50,
|
||||||
) {
|
MODBUS_TIMEOUT_MIN: 100,
|
||||||
if (
|
MODBUS_TIMEOUT_MAX: 20000,
|
||||||
value &&
|
SYSLOG_MARK_INTERVAL_MIN: 0,
|
||||||
(value === 1 ||
|
SYSLOG_MARK_INTERVAL_MAX: 3600,
|
||||||
(value >= 6 && value <= 11) ||
|
SHOWER_MIN_DURATION_MIN: 10,
|
||||||
(value >= 16 && value <= 17) ||
|
SHOWER_MIN_DURATION_MAX: 360,
|
||||||
value === 20 ||
|
SHOWER_ALERT_TRIGGER_MIN: 1,
|
||||||
value === 24 ||
|
SHOWER_ALERT_TRIGGER_MAX: 20,
|
||||||
(value >= 28 && value <= 31) ||
|
SHOWER_ALERT_COLDSHOT_MIN: 1,
|
||||||
value > 40 ||
|
SHOWER_ALERT_COLDSHOT_MAX: 10,
|
||||||
value < 0)
|
REMOTE_TIMEOUT_MIN: 1,
|
||||||
) {
|
REMOTE_TIMEOUT_MAX: 240,
|
||||||
callback('Must be an valid GPIO port');
|
OFFSET_MIN: 0,
|
||||||
} else {
|
OFFSET_MAX: 255,
|
||||||
callback();
|
COMMAND_MIN: 1,
|
||||||
}
|
COMMAND_MAX: 300,
|
||||||
}
|
NAME_MAX_LENGTH: 19,
|
||||||
};
|
HEX_BASE: 16
|
||||||
|
} as const;
|
||||||
|
|
||||||
export const GPIO_VALIDATORC3 = {
|
type ValidationRules = Array<{
|
||||||
validator(
|
required?: boolean;
|
||||||
_rule: InternalRuleItem,
|
message?: string;
|
||||||
value: number,
|
[key: string]: unknown;
|
||||||
callback: (error?: string) => void
|
}>;
|
||||||
) {
|
|
||||||
if (value && ((value >= 11 && value <= 19) || value > 21 || value < 0)) {
|
|
||||||
callback('Must be an valid GPIO port');
|
|
||||||
} else {
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const GPIO_VALIDATORS2 = {
|
export const createSettingsValidator = (settings: Settings) => {
|
||||||
validator(
|
const schema: Record<string, ValidationRules> = {};
|
||||||
_rule: InternalRuleItem,
|
|
||||||
value: number,
|
|
||||||
callback: (error?: string) => void
|
|
||||||
) {
|
|
||||||
if (
|
|
||||||
value &&
|
|
||||||
((value >= 19 && value <= 20) ||
|
|
||||||
(value >= 22 && value <= 32) ||
|
|
||||||
value > 40 ||
|
|
||||||
value < 0)
|
|
||||||
) {
|
|
||||||
callback('Must be an valid GPIO port');
|
|
||||||
} else {
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const GPIO_VALIDATORS3 = {
|
// Syslog validations
|
||||||
validator(
|
if (settings.syslog_enabled) {
|
||||||
_rule: InternalRuleItem,
|
schema.syslog_host = [
|
||||||
value: number,
|
|
||||||
callback: (error?: string) => void
|
|
||||||
) {
|
|
||||||
if (
|
|
||||||
value &&
|
|
||||||
((value >= 19 && value <= 20) ||
|
|
||||||
(value >= 22 && value <= 37) ||
|
|
||||||
(value >= 39 && value <= 42) ||
|
|
||||||
value > 48 ||
|
|
||||||
value < 0)
|
|
||||||
) {
|
|
||||||
callback('Must be an valid GPIO port');
|
|
||||||
} else {
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const createSettingsValidator = (settings: Settings) =>
|
|
||||||
new Schema({
|
|
||||||
...(settings.board_profile === 'CUSTOM' &&
|
|
||||||
settings.platform === 'ESP32' && {
|
|
||||||
led_gpio: [
|
|
||||||
{ required: true, message: 'LED GPIO is required' },
|
|
||||||
GPIO_VALIDATOR
|
|
||||||
],
|
|
||||||
dallas_gpio: [
|
|
||||||
{ required: true, message: 'GPIO is required' },
|
|
||||||
GPIO_VALIDATOR
|
|
||||||
],
|
|
||||||
pbutton_gpio: [
|
|
||||||
{ required: true, message: 'Button GPIO is required' },
|
|
||||||
GPIO_VALIDATOR
|
|
||||||
],
|
|
||||||
tx_gpio: [
|
|
||||||
{ required: true, message: 'Tx GPIO is required' },
|
|
||||||
GPIO_VALIDATOR
|
|
||||||
],
|
|
||||||
rx_gpio: [{ required: true, message: 'Rx GPIO is required' }, GPIO_VALIDATOR]
|
|
||||||
}),
|
|
||||||
...(settings.board_profile === 'CUSTOM' &&
|
|
||||||
settings.platform === 'ESP32C3' && {
|
|
||||||
led_gpio: [
|
|
||||||
{ required: true, message: 'LED GPIO is required' },
|
|
||||||
GPIO_VALIDATORC3
|
|
||||||
],
|
|
||||||
dallas_gpio: [
|
|
||||||
{ required: true, message: 'GPIO is required' },
|
|
||||||
GPIO_VALIDATORC3
|
|
||||||
],
|
|
||||||
pbutton_gpio: [
|
|
||||||
{ required: true, message: 'Button GPIO is required' },
|
|
||||||
GPIO_VALIDATORC3
|
|
||||||
],
|
|
||||||
tx_gpio: [
|
|
||||||
{ required: true, message: 'Tx GPIO is required' },
|
|
||||||
GPIO_VALIDATORC3
|
|
||||||
],
|
|
||||||
rx_gpio: [
|
|
||||||
{ required: true, message: 'Rx GPIO is required' },
|
|
||||||
GPIO_VALIDATORC3
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
...(settings.board_profile === 'CUSTOM' &&
|
|
||||||
settings.platform === 'ESP32S2' && {
|
|
||||||
led_gpio: [
|
|
||||||
{ required: true, message: 'LED GPIO is required' },
|
|
||||||
GPIO_VALIDATORS2
|
|
||||||
],
|
|
||||||
dallas_gpio: [
|
|
||||||
{ required: true, message: 'GPIO is required' },
|
|
||||||
GPIO_VALIDATORS2
|
|
||||||
],
|
|
||||||
pbutton_gpio: [
|
|
||||||
{ required: true, message: 'Button GPIO is required' },
|
|
||||||
GPIO_VALIDATORS2
|
|
||||||
],
|
|
||||||
tx_gpio: [
|
|
||||||
{ required: true, message: 'Tx GPIO is required' },
|
|
||||||
GPIO_VALIDATORS2
|
|
||||||
],
|
|
||||||
rx_gpio: [
|
|
||||||
{ required: true, message: 'Rx GPIO is required' },
|
|
||||||
GPIO_VALIDATORS2
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
...(settings.board_profile === 'CUSTOM' &&
|
|
||||||
settings.platform === 'ESP32S3' && {
|
|
||||||
led_gpio: [
|
|
||||||
{ required: true, message: 'LED GPIO is required' },
|
|
||||||
GPIO_VALIDATORS3
|
|
||||||
],
|
|
||||||
dallas_gpio: [
|
|
||||||
{ required: true, message: 'GPIO is required' },
|
|
||||||
GPIO_VALIDATORS3
|
|
||||||
],
|
|
||||||
pbutton_gpio: [
|
|
||||||
{ required: true, message: 'Button GPIO is required' },
|
|
||||||
GPIO_VALIDATORS3
|
|
||||||
],
|
|
||||||
tx_gpio: [
|
|
||||||
{ required: true, message: 'Tx GPIO is required' },
|
|
||||||
GPIO_VALIDATORS3
|
|
||||||
],
|
|
||||||
rx_gpio: [
|
|
||||||
{ required: true, message: 'Rx GPIO is required' },
|
|
||||||
GPIO_VALIDATORS3
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
...(settings.syslog_enabled && {
|
|
||||||
syslog_host: [
|
|
||||||
{ required: true, message: 'Host is required' },
|
{ required: true, message: 'Host is required' },
|
||||||
IP_OR_HOSTNAME_VALIDATOR
|
IP_OR_HOSTNAME_VALIDATOR
|
||||||
],
|
];
|
||||||
syslog_port: [
|
schema.syslog_port = [
|
||||||
{ required: true, message: 'Port is required' },
|
{ required: true, message: 'Port is required' },
|
||||||
{ type: 'number', min: 0, max: 65535, message: 'Invalid Port' }
|
{
|
||||||
],
|
type: 'number',
|
||||||
syslog_mark_interval: [
|
min: VALIDATION_LIMITS.PORT_MIN,
|
||||||
|
max: VALIDATION_LIMITS.PORT_MAX,
|
||||||
|
message: 'Invalid Port'
|
||||||
|
}
|
||||||
|
];
|
||||||
|
schema.syslog_mark_interval = [
|
||||||
{ required: true, message: 'Mark interval is required' },
|
{ required: true, message: 'Mark interval is required' },
|
||||||
{ type: 'number', min: 0, max: 10, message: 'Must be between 0 and 10' }
|
{
|
||||||
]
|
type: 'number',
|
||||||
}),
|
min: VALIDATION_LIMITS.SYSLOG_MARK_INTERVAL_MIN,
|
||||||
...(settings.modbus_enabled && {
|
max: VALIDATION_LIMITS.SYSLOG_MARK_INTERVAL_MAX,
|
||||||
modbus_max_clients: [
|
message: `Must be between ${VALIDATION_LIMITS.SYSLOG_MARK_INTERVAL_MIN} and ${VALIDATION_LIMITS.SYSLOG_MARK_INTERVAL_MAX}`
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Modbus validations
|
||||||
|
if (settings.modbus_enabled) {
|
||||||
|
schema.modbus_max_clients = [
|
||||||
{ required: true, message: 'Max clients is required' },
|
{ required: true, message: 'Max clients is required' },
|
||||||
{ type: 'number', min: 0, max: 50, message: 'Invalid number' }
|
{
|
||||||
],
|
type: 'number',
|
||||||
modbus_port: [
|
min: VALIDATION_LIMITS.MODBUS_MAX_CLIENTS_MIN,
|
||||||
|
max: VALIDATION_LIMITS.MODBUS_MAX_CLIENTS_MAX,
|
||||||
|
message: 'Invalid number'
|
||||||
|
}
|
||||||
|
];
|
||||||
|
schema.modbus_port = [
|
||||||
{ required: true, message: 'Port is required' },
|
{ required: true, message: 'Port is required' },
|
||||||
{ type: 'number', min: 0, max: 65535, message: 'Invalid Port' }
|
{
|
||||||
],
|
type: 'number',
|
||||||
modbus_timeout: [
|
min: VALIDATION_LIMITS.PORT_MIN,
|
||||||
|
max: VALIDATION_LIMITS.PORT_MAX,
|
||||||
|
message: 'Invalid Port'
|
||||||
|
}
|
||||||
|
];
|
||||||
|
schema.modbus_timeout = [
|
||||||
{ required: true, message: 'Timeout is required' },
|
{ required: true, message: 'Timeout is required' },
|
||||||
{
|
{
|
||||||
type: 'number',
|
type: 'number',
|
||||||
min: 100,
|
min: VALIDATION_LIMITS.MODBUS_TIMEOUT_MIN,
|
||||||
max: 20000,
|
max: VALIDATION_LIMITS.MODBUS_TIMEOUT_MAX,
|
||||||
message: 'Must be between 100 and 20000'
|
message: `Must be between ${VALIDATION_LIMITS.MODBUS_TIMEOUT_MIN} and ${VALIDATION_LIMITS.MODBUS_TIMEOUT_MAX}`
|
||||||
}
|
}
|
||||||
]
|
];
|
||||||
}),
|
|
||||||
...(settings.shower_timer && {
|
|
||||||
shower_min_duration: [
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: 10,
|
|
||||||
max: 360,
|
|
||||||
message: 'Time must be between 10 and 360 seconds'
|
|
||||||
}
|
}
|
||||||
]
|
|
||||||
}),
|
|
||||||
...(settings.shower_alert && {
|
|
||||||
shower_alert_trigger: [
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: 1,
|
|
||||||
max: 20,
|
|
||||||
message: 'Time must be between 1 and 20 minutes'
|
|
||||||
}
|
|
||||||
],
|
|
||||||
shower_alert_coldshot: [
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: 1,
|
|
||||||
max: 10,
|
|
||||||
message: 'Time must be between 1 and 10 seconds'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}),
|
|
||||||
...(settings.remote_timeout_en && {
|
|
||||||
remote_timeout: [
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: 1,
|
|
||||||
max: 240,
|
|
||||||
message: 'Timeout must be between 1 and 240 hours'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
})
|
|
||||||
});
|
|
||||||
|
|
||||||
export const uniqueNameValidator = (schedule: ScheduleItem[], o_name?: string) => ({
|
// Shower timer validations
|
||||||
|
if (settings.shower_timer) {
|
||||||
|
schema.shower_min_duration = [
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: VALIDATION_LIMITS.SHOWER_MIN_DURATION_MIN,
|
||||||
|
max: VALIDATION_LIMITS.SHOWER_MIN_DURATION_MAX,
|
||||||
|
message: `Time must be between ${VALIDATION_LIMITS.SHOWER_MIN_DURATION_MIN} and ${VALIDATION_LIMITS.SHOWER_MIN_DURATION_MAX} seconds`
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Shower alert validations
|
||||||
|
if (settings.shower_alert) {
|
||||||
|
schema.shower_alert_trigger = [
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: VALIDATION_LIMITS.SHOWER_ALERT_TRIGGER_MIN,
|
||||||
|
max: VALIDATION_LIMITS.SHOWER_ALERT_TRIGGER_MAX,
|
||||||
|
message: `Time must be between ${VALIDATION_LIMITS.SHOWER_ALERT_TRIGGER_MIN} and ${VALIDATION_LIMITS.SHOWER_ALERT_TRIGGER_MAX} minutes`
|
||||||
|
}
|
||||||
|
];
|
||||||
|
schema.shower_alert_coldshot = [
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: VALIDATION_LIMITS.SHOWER_ALERT_COLDSHOT_MIN,
|
||||||
|
max: VALIDATION_LIMITS.SHOWER_ALERT_COLDSHOT_MAX,
|
||||||
|
message: `Time must be between ${VALIDATION_LIMITS.SHOWER_ALERT_COLDSHOT_MIN} and ${VALIDATION_LIMITS.SHOWER_ALERT_COLDSHOT_MAX} seconds`
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remote timeout validations
|
||||||
|
if (settings.remote_timeout_en) {
|
||||||
|
schema.remote_timeout = [
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: VALIDATION_LIMITS.REMOTE_TIMEOUT_MIN,
|
||||||
|
max: VALIDATION_LIMITS.REMOTE_TIMEOUT_MAX,
|
||||||
|
message: `Timeout must be between ${VALIDATION_LIMITS.REMOTE_TIMEOUT_MIN} and ${VALIDATION_LIMITS.REMOTE_TIMEOUT_MAX} hours`
|
||||||
|
}
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
return new Schema(schema);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Generic unique name validator factory
|
||||||
|
const createUniqueNameValidator = <T extends { name: string }>(
|
||||||
|
items: T[],
|
||||||
|
originalName?: string
|
||||||
|
) => ({
|
||||||
validator(
|
validator(
|
||||||
_rule: InternalRuleItem,
|
_rule: InternalRuleItem,
|
||||||
name: string,
|
name: string,
|
||||||
@@ -285,43 +170,22 @@ export const uniqueNameValidator = (schedule: ScheduleItem[], o_name?: string) =
|
|||||||
) {
|
) {
|
||||||
if (
|
if (
|
||||||
name !== '' &&
|
name !== '' &&
|
||||||
(o_name === undefined || o_name.toLowerCase() !== name.toLowerCase()) &&
|
(originalName === undefined ||
|
||||||
schedule.find((si) => si.name.toLowerCase() === name.toLowerCase())
|
originalName.toLowerCase() !== name.toLowerCase()) &&
|
||||||
|
items.find((item) => item.name.toLowerCase() === name.toLowerCase())
|
||||||
) {
|
) {
|
||||||
callback('Name already in use');
|
callback(ERROR_MESSAGES.NAME_DUPLICATE);
|
||||||
} else {
|
return;
|
||||||
|
}
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
export const schedulerItemValidation = (
|
// Generic field name validator (for cases where the name field has different property names)
|
||||||
schedule: ScheduleItem[],
|
const createUniqueFieldNameValidator = <T>(
|
||||||
scheduleItem: ScheduleItem
|
items: T[],
|
||||||
) =>
|
getName: (item: T) => string,
|
||||||
new Schema({
|
originalName?: string
|
||||||
name: [
|
|
||||||
{
|
|
||||||
type: 'string',
|
|
||||||
pattern: /^[a-zA-Z0-9_]{0,19}$/,
|
|
||||||
message: "Must be <20 characters: alphanumeric or '_'"
|
|
||||||
},
|
|
||||||
...[uniqueNameValidator(schedule, scheduleItem.o_name)]
|
|
||||||
],
|
|
||||||
cmd: [
|
|
||||||
{ required: true, message: 'Command is required' },
|
|
||||||
{
|
|
||||||
type: 'string',
|
|
||||||
min: 1,
|
|
||||||
max: 300,
|
|
||||||
message: 'Command must be 1-300 characters'
|
|
||||||
}
|
|
||||||
]
|
|
||||||
});
|
|
||||||
|
|
||||||
export const uniqueCustomNameValidator = (
|
|
||||||
entity: EntityItem[],
|
|
||||||
o_name?: string
|
|
||||||
) => ({
|
) => ({
|
||||||
validator(
|
validator(
|
||||||
_rule: InternalRuleItem,
|
_rule: InternalRuleItem,
|
||||||
@@ -329,58 +193,91 @@ export const uniqueCustomNameValidator = (
|
|||||||
callback: (error?: string) => void
|
callback: (error?: string) => void
|
||||||
) {
|
) {
|
||||||
if (
|
if (
|
||||||
(o_name === undefined || o_name.toLowerCase() !== name.toLowerCase()) &&
|
name !== '' &&
|
||||||
entity.find((ei) => ei.name.toLowerCase() === name.toLowerCase())
|
(originalName === undefined ||
|
||||||
|
originalName.toLowerCase() !== name.toLowerCase()) &&
|
||||||
|
items.find((item) => getName(item).toLowerCase() === name.toLowerCase())
|
||||||
) {
|
) {
|
||||||
callback('Name already in use');
|
callback(ERROR_MESSAGES.NAME_DUPLICATE);
|
||||||
} else {
|
return;
|
||||||
|
}
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const NAME_PATTERN_BASE = '[a-zA-Z0-9_]';
|
||||||
|
const NAME_PATTERN_MESSAGE = `Must be <${VALIDATION_LIMITS.NAME_MAX_LENGTH + 1} characters: alphanumeric or '_'`;
|
||||||
|
|
||||||
|
const NAME_PATTERN = {
|
||||||
|
type: 'string' as const,
|
||||||
|
pattern: new RegExp(
|
||||||
|
`^${NAME_PATTERN_BASE}{0,${VALIDATION_LIMITS.NAME_MAX_LENGTH}}$`
|
||||||
|
),
|
||||||
|
message: NAME_PATTERN_MESSAGE
|
||||||
|
};
|
||||||
|
|
||||||
|
const NAME_PATTERN_REQUIRED = {
|
||||||
|
type: 'string' as const,
|
||||||
|
pattern: new RegExp(
|
||||||
|
`^${NAME_PATTERN_BASE}{1,${VALIDATION_LIMITS.NAME_MAX_LENGTH}}$`
|
||||||
|
),
|
||||||
|
message: NAME_PATTERN_MESSAGE
|
||||||
|
};
|
||||||
|
|
||||||
|
export const uniqueNameValidator = (schedule: ScheduleItem[], o_name?: string) =>
|
||||||
|
createUniqueNameValidator(schedule, o_name);
|
||||||
|
|
||||||
|
export const schedulerItemValidation = (
|
||||||
|
schedule: ScheduleItem[],
|
||||||
|
scheduleItem: ScheduleItem
|
||||||
|
) =>
|
||||||
|
new Schema({
|
||||||
|
name: [NAME_PATTERN, uniqueNameValidator(schedule, scheduleItem.o_name)],
|
||||||
|
cmd: [
|
||||||
|
{ required: true, message: 'Command is required' },
|
||||||
|
{
|
||||||
|
type: 'string',
|
||||||
|
min: VALIDATION_LIMITS.COMMAND_MIN,
|
||||||
|
max: VALIDATION_LIMITS.COMMAND_MAX,
|
||||||
|
message: `Command must be ${VALIDATION_LIMITS.COMMAND_MIN}-${VALIDATION_LIMITS.COMMAND_MAX} characters`
|
||||||
|
}
|
||||||
|
]
|
||||||
|
});
|
||||||
|
|
||||||
|
export const uniqueCustomNameValidator = (entity: EntityItem[], o_name?: string) =>
|
||||||
|
createUniqueNameValidator(entity, o_name);
|
||||||
|
|
||||||
|
const hexValidator = {
|
||||||
|
validator(
|
||||||
|
_rule: InternalRuleItem,
|
||||||
|
value: string,
|
||||||
|
callback: (error?: string) => void
|
||||||
|
) {
|
||||||
|
if (!value || Number.isNaN(Number.parseInt(value, VALIDATION_LIMITS.HEX_BASE))) {
|
||||||
|
callback(ERROR_MESSAGES.HEX_REQUIRED);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
export const entityItemValidation = (entity: EntityItem[], entityItem: EntityItem) =>
|
export const entityItemValidation = (entity: EntityItem[], entityItem: EntityItem) =>
|
||||||
new Schema({
|
new Schema({
|
||||||
name: [
|
name: [
|
||||||
{ required: true, message: 'Name is required' },
|
{ required: true, message: 'Name is required' },
|
||||||
{
|
NAME_PATTERN_REQUIRED,
|
||||||
type: 'string',
|
uniqueCustomNameValidator(entity, entityItem.o_name)
|
||||||
pattern: /^[a-zA-Z0-9_]{1,19}$/,
|
|
||||||
message: "Must be <20 characters: alphanumeric or '_'"
|
|
||||||
},
|
|
||||||
...[uniqueCustomNameValidator(entity, entityItem.o_name)]
|
|
||||||
],
|
|
||||||
device_id: [
|
|
||||||
{
|
|
||||||
validator(
|
|
||||||
_rule: InternalRuleItem,
|
|
||||||
value: string,
|
|
||||||
callback: (error?: string) => void
|
|
||||||
) {
|
|
||||||
if (isNaN(parseInt(value, 16))) {
|
|
||||||
callback('Is required and must be in hex format');
|
|
||||||
}
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
],
|
|
||||||
type_id: [
|
|
||||||
{
|
|
||||||
validator(
|
|
||||||
_rule: InternalRuleItem,
|
|
||||||
value: string,
|
|
||||||
callback: (error?: string) => void
|
|
||||||
) {
|
|
||||||
if (isNaN(parseInt(value, 16))) {
|
|
||||||
callback('Is required and must be in hex format');
|
|
||||||
}
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
],
|
],
|
||||||
|
device_id: [hexValidator],
|
||||||
|
type_id: [hexValidator],
|
||||||
offset: [
|
offset: [
|
||||||
{ required: true, message: 'Offset is required' },
|
{ required: true, message: 'Offset is required' },
|
||||||
{ type: 'number', min: 0, max: 255, message: 'Must be between 0 and 255' }
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: VALIDATION_LIMITS.OFFSET_MIN,
|
||||||
|
max: VALIDATION_LIMITS.OFFSET_MAX,
|
||||||
|
message: `Must be between ${VALIDATION_LIMITS.OFFSET_MIN} and ${VALIDATION_LIMITS.OFFSET_MAX}`
|
||||||
|
}
|
||||||
],
|
],
|
||||||
factor: [{ required: true, message: 'is required' }]
|
factor: [{ required: true, message: 'is required' }]
|
||||||
});
|
});
|
||||||
@@ -388,93 +285,34 @@ export const entityItemValidation = (entity: EntityItem[], entityItem: EntityIte
|
|||||||
export const uniqueTemperatureNameValidator = (
|
export const uniqueTemperatureNameValidator = (
|
||||||
sensors: TemperatureSensor[],
|
sensors: TemperatureSensor[],
|
||||||
o_name?: string
|
o_name?: string
|
||||||
) => ({
|
) => createUniqueFieldNameValidator(sensors, (s) => s.n, o_name);
|
||||||
validator(_rule: InternalRuleItem, n: string, callback: (error?: string) => void) {
|
|
||||||
if (
|
|
||||||
(o_name === undefined || o_name.toLowerCase() !== n.toLowerCase()) &&
|
|
||||||
n !== '' &&
|
|
||||||
sensors.find((ts) => ts.n.toLowerCase() === n.toLowerCase())
|
|
||||||
) {
|
|
||||||
callback('Name already in use');
|
|
||||||
} else {
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
export const temperatureSensorItemValidation = (
|
export const temperatureSensorItemValidation = (
|
||||||
sensors: TemperatureSensor[],
|
sensors: TemperatureSensor[],
|
||||||
sensor: TemperatureSensor
|
sensor: TemperatureSensor
|
||||||
) =>
|
) =>
|
||||||
new Schema({
|
new Schema({
|
||||||
n: [
|
n: [NAME_PATTERN, uniqueTemperatureNameValidator(sensors, sensor.o_n)]
|
||||||
{
|
|
||||||
type: 'string',
|
|
||||||
pattern: /^[a-zA-Z0-9_]{0,19}$/,
|
|
||||||
message: "Must be <20 characters: alphanumeric or '_'"
|
|
||||||
},
|
|
||||||
...[uniqueTemperatureNameValidator(sensors, sensor.o_n)]
|
|
||||||
]
|
|
||||||
});
|
|
||||||
|
|
||||||
export const isGPIOUniqueValidator = (sensors: AnalogSensor[]) => ({
|
|
||||||
validator(
|
|
||||||
_rule: InternalRuleItem,
|
|
||||||
gpio: number,
|
|
||||||
callback: (error?: string) => void
|
|
||||||
) {
|
|
||||||
if (sensors.find((as) => as.g === gpio)) {
|
|
||||||
callback('GPIO already in use');
|
|
||||||
} else {
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
export const uniqueAnalogNameValidator = (
|
export const uniqueAnalogNameValidator = (
|
||||||
sensors: AnalogSensor[],
|
sensors: AnalogSensor[],
|
||||||
o_name?: string
|
o_name?: string
|
||||||
) => ({
|
) => createUniqueFieldNameValidator(sensors, (s) => s.n, o_name);
|
||||||
validator(_rule: InternalRuleItem, n: string, callback: (error?: string) => void) {
|
|
||||||
if (
|
|
||||||
(o_name === undefined || o_name.toLowerCase() !== n.toLowerCase()) &&
|
|
||||||
n !== '' &&
|
|
||||||
sensors.find((as) => as.n.toLowerCase() === n.toLowerCase())
|
|
||||||
) {
|
|
||||||
callback('Name already in use');
|
|
||||||
} else {
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
export const analogSensorItemValidation = (
|
export const analogSensorItemValidation = (
|
||||||
sensors: AnalogSensor[],
|
sensors: AnalogSensor[],
|
||||||
sensor: AnalogSensor,
|
sensor: AnalogSensor
|
||||||
creating: boolean,
|
) => {
|
||||||
platform: string
|
return new Schema({
|
||||||
) =>
|
// name is required and must be unique
|
||||||
new Schema({
|
|
||||||
n: [
|
n: [
|
||||||
{
|
{ required: true, message: 'Name is required' },
|
||||||
type: 'string',
|
NAME_PATTERN,
|
||||||
pattern: /^[a-zA-Z0-9_]{0,19}$/,
|
uniqueAnalogNameValidator(sensors, sensor.o_n)
|
||||||
message: "Must be <20 characters: alphanumeric or '_'"
|
|
||||||
},
|
|
||||||
...[uniqueAnalogNameValidator(sensors, sensor.o_n)]
|
|
||||||
],
|
|
||||||
g: [
|
|
||||||
{ required: true, message: 'GPIO is required' },
|
|
||||||
platform === 'ESP32S3'
|
|
||||||
? GPIO_VALIDATORS3
|
|
||||||
: platform === 'ESP32S2'
|
|
||||||
? GPIO_VALIDATORS2
|
|
||||||
: platform === 'ESP32C3'
|
|
||||||
? GPIO_VALIDATORC3
|
|
||||||
: GPIO_VALIDATOR,
|
|
||||||
...(creating ? [isGPIOUniqueValidator(sensors)] : [])
|
|
||||||
]
|
]
|
||||||
});
|
});
|
||||||
|
};
|
||||||
|
|
||||||
export const deviceValueItemValidation = (dv: DeviceValue) =>
|
export const deviceValueItemValidation = (dv: DeviceValue) =>
|
||||||
new Schema({
|
new Schema({
|
||||||
@@ -488,11 +326,12 @@ export const deviceValueItemValidation = (dv: DeviceValue) =>
|
|||||||
) {
|
) {
|
||||||
if (
|
if (
|
||||||
typeof value === 'number' &&
|
typeof value === 'number' &&
|
||||||
dv.m &&
|
dv.m !== undefined &&
|
||||||
dv.x &&
|
dv.x !== undefined &&
|
||||||
(value < dv.m || value > dv.x)
|
(value < dv.m || value > dv.x)
|
||||||
) {
|
) {
|
||||||
callback('Value out of range');
|
callback(ERROR_MESSAGES.VALUE_OUT_OF_RANGE);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
@@ -27,6 +27,19 @@ export const isAPEnabled = ({ provision_mode }: APSettingsType) =>
|
|||||||
provision_mode === APProvisionMode.AP_MODE_ALWAYS ||
|
provision_mode === APProvisionMode.AP_MODE_ALWAYS ||
|
||||||
provision_mode === APProvisionMode.AP_MODE_DISCONNECTED;
|
provision_mode === APProvisionMode.AP_MODE_DISCONNECTED;
|
||||||
|
|
||||||
|
// Efficient range function without recursion
|
||||||
|
const createRange = (start: number, end: number): number[] => {
|
||||||
|
const result: number[] = [];
|
||||||
|
for (let i = start; i <= end; i++) {
|
||||||
|
result.push(i);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Pre-computed ranges for better performance
|
||||||
|
const CHANNEL_RANGE = createRange(1, 14);
|
||||||
|
const MAX_CLIENTS_RANGE = createRange(1, 9);
|
||||||
|
|
||||||
const APSettings = () => {
|
const APSettings = () => {
|
||||||
const {
|
const {
|
||||||
loadData,
|
loadData,
|
||||||
@@ -50,19 +63,24 @@ const APSettings = () => {
|
|||||||
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
const updateFormValue = useMemo(
|
||||||
origData,
|
() =>
|
||||||
|
updateValueDirty(
|
||||||
|
origData as unknown as Record<string, unknown>,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue as (value: unknown) => void
|
||||||
|
),
|
||||||
|
[origData, dirtyFlags, setDirtyFlags, updateDataValue]
|
||||||
);
|
);
|
||||||
|
|
||||||
const content = () => {
|
// Memoize AP enabled state
|
||||||
if (!data) {
|
const apEnabled = useMemo(() => (data ? isAPEnabled(data) : false), [data]);
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
||||||
}
|
// Memoize validation and submit handler
|
||||||
|
const validateAndSubmit = useCallback(async () => {
|
||||||
|
if (!data) return;
|
||||||
|
|
||||||
const validateAndSubmit = async () => {
|
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(createAPSettingsValidator(data), data);
|
await validate(createAPSettingsValidator(data), data);
|
||||||
@@ -70,11 +88,11 @@ const APSettings = () => {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
}, [data, saveData]);
|
||||||
|
|
||||||
// no lodash - https://asleepace.com/blog/typescript-range-without-a-loop/
|
const content = () => {
|
||||||
function range(a: number, b: number): number[] {
|
if (!data) {
|
||||||
return a < b ? [a, ...range(a + 1, b)] : [b];
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -100,7 +118,7 @@ const APSettings = () => {
|
|||||||
{LL.AP_PROVIDE_TEXT_3()}
|
{LL.AP_PROVIDE_TEXT_3()}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
</ValidatedTextField>
|
</ValidatedTextField>
|
||||||
{isAPEnabled(data) && (
|
{apEnabled && (
|
||||||
<>
|
<>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors || {}}
|
||||||
@@ -134,7 +152,7 @@ const APSettings = () => {
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
>
|
>
|
||||||
{range(1, 14).map((i) => (
|
{CHANNEL_RANGE.map((i) => (
|
||||||
<MenuItem key={i} value={i}>
|
<MenuItem key={i} value={i}>
|
||||||
{i}
|
{i}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
@@ -162,7 +180,7 @@ const APSettings = () => {
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
>
|
>
|
||||||
{range(1, 9).map((i) => (
|
{MAX_CLIENTS_RANGE.map((i) => (
|
||||||
<MenuItem key={i} value={i}>
|
<MenuItem key={i} value={i}>
|
||||||
{i}
|
{i}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -28,6 +28,7 @@ import {
|
|||||||
FormLoader,
|
FormLoader,
|
||||||
MessageBox,
|
MessageBox,
|
||||||
SectionContent,
|
SectionContent,
|
||||||
|
ValidatedPasswordField,
|
||||||
ValidatedTextField,
|
ValidatedTextField,
|
||||||
useLayoutTitle
|
useLayoutTitle
|
||||||
} from 'components';
|
} from 'components';
|
||||||
@@ -37,13 +38,13 @@ import { validate } from 'validators';
|
|||||||
|
|
||||||
import { API, getBoardProfile, readSettings, writeSettings } from '../../api/app';
|
import { API, getBoardProfile, readSettings, writeSettings } from '../../api/app';
|
||||||
import { BOARD_PROFILES } from '../main/types';
|
import { BOARD_PROFILES } from '../main/types';
|
||||||
import type { APIcall, Settings } from '../main/types';
|
import type { APIcall, BoardProfileKey, Settings } from '../main/types';
|
||||||
import { createSettingsValidator } from '../main/validators';
|
import { createSettingsValidator } from '../main/validators';
|
||||||
|
|
||||||
export function boardProfileSelectItems() {
|
export function boardProfileSelectItems() {
|
||||||
return Object.keys(BOARD_PROFILES).map((code) => (
|
return Object.keys(BOARD_PROFILES).map((code) => (
|
||||||
<MenuItem key={code} value={code}>
|
<MenuItem key={code} value={code}>
|
||||||
{BOARD_PROFILES[code]}
|
{BOARD_PROFILES[code as BoardProfileKey]}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
@@ -72,7 +73,7 @@ const ApplicationSettings = () => {
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
const updateFormValue = updateValueDirty(
|
||||||
origData,
|
origData as unknown as Record<string, unknown>,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue as (value: unknown) => void
|
||||||
@@ -106,39 +107,49 @@ const ApplicationSettings = () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const doRestart = async () => {
|
// Memoized input props to prevent recreation on every render
|
||||||
|
const SecondsInputProps = useMemo(
|
||||||
|
() => ({
|
||||||
|
endAdornment: <InputAdornment position="end">{LL.SECONDS()}</InputAdornment>
|
||||||
|
}),
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const MinutesInputProps = useMemo(
|
||||||
|
() => ({
|
||||||
|
endAdornment: <InputAdornment position="end">{LL.MINUTES()}</InputAdornment>
|
||||||
|
}),
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const HoursInputProps = useMemo(
|
||||||
|
() => ({
|
||||||
|
endAdornment: <InputAdornment position="end">{LL.HOURS()}</InputAdornment>
|
||||||
|
}),
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const doRestart = useCallback(async () => {
|
||||||
setRestarting(true);
|
setRestarting(true);
|
||||||
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
||||||
(error: Error) => {
|
(error: Error) => {
|
||||||
toast.error(error.message);
|
toast.error(error.message);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
};
|
}, [sendAPI]);
|
||||||
|
|
||||||
const updateBoardProfile = async (board_profile: string) => {
|
const updateBoardProfile = useCallback(
|
||||||
|
async (board_profile: string) => {
|
||||||
await readBoardProfile(board_profile).catch((error: Error) => {
|
await readBoardProfile(board_profile).catch((error: Error) => {
|
||||||
toast.error(error.message);
|
toast.error(error.message);
|
||||||
});
|
});
|
||||||
};
|
},
|
||||||
|
[readBoardProfile]
|
||||||
|
);
|
||||||
|
|
||||||
useLayoutTitle(LL.APPLICATION());
|
useLayoutTitle(LL.APPLICATION());
|
||||||
|
|
||||||
const SecondsInputProps = {
|
const validateAndSubmit = useCallback(async () => {
|
||||||
endAdornment: <InputAdornment position="end">{LL.SECONDS()}</InputAdornment>
|
|
||||||
};
|
|
||||||
const MinutesInputProps = {
|
|
||||||
endAdornment: <InputAdornment position="end">{LL.MINUTES()}</InputAdornment>
|
|
||||||
};
|
|
||||||
const HoursInputProps = {
|
|
||||||
endAdornment: <InputAdornment position="end">{LL.HOURS()}</InputAdornment>
|
|
||||||
};
|
|
||||||
|
|
||||||
const content = () => {
|
|
||||||
if (!data || !hardwareData) {
|
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
const validateAndSubmit = async () => {
|
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(createSettingsValidator(data), data);
|
await validate(createSettingsValidator(data), data);
|
||||||
@@ -147,9 +158,10 @@ const ApplicationSettings = () => {
|
|||||||
} finally {
|
} finally {
|
||||||
await saveData();
|
await saveData();
|
||||||
}
|
}
|
||||||
};
|
}, [data, saveData]);
|
||||||
|
|
||||||
const changeBoardProfile = (event: React.ChangeEvent<HTMLInputElement>) => {
|
const changeBoardProfile = useCallback(
|
||||||
|
(event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const boardProfile = event.target.value;
|
const boardProfile = event.target.value;
|
||||||
updateFormValue(event);
|
updateFormValue(event);
|
||||||
if (boardProfile === 'CUSTOM') {
|
if (boardProfile === 'CUSTOM') {
|
||||||
@@ -160,12 +172,22 @@ const ApplicationSettings = () => {
|
|||||||
} else {
|
} else {
|
||||||
void updateBoardProfile(boardProfile);
|
void updateBoardProfile(boardProfile);
|
||||||
}
|
}
|
||||||
};
|
},
|
||||||
|
[data, updateBoardProfile, updateFormValue, updateDataValue]
|
||||||
|
);
|
||||||
|
|
||||||
const restart = async () => {
|
const restart = useCallback(async () => {
|
||||||
await validateAndSubmit();
|
await validateAndSubmit();
|
||||||
await doRestart();
|
await doRestart();
|
||||||
};
|
}, [validateAndSubmit, doRestart]);
|
||||||
|
|
||||||
|
// Memoize board profile select items to prevent recreation
|
||||||
|
const boardProfileItems = useMemo(() => boardProfileSelectItems(), []);
|
||||||
|
|
||||||
|
const content = () => {
|
||||||
|
if (!data || !hardwareData) {
|
||||||
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -307,9 +329,9 @@ const ApplicationSettings = () => {
|
|||||||
>
|
>
|
||||||
<MenuItem value={-1}>OFF</MenuItem>
|
<MenuItem value={-1}>OFF</MenuItem>
|
||||||
<MenuItem value={3}>ERR</MenuItem>
|
<MenuItem value={3}>ERR</MenuItem>
|
||||||
|
<MenuItem value={4}>WARN</MenuItem>
|
||||||
<MenuItem value={5}>NOTICE</MenuItem>
|
<MenuItem value={5}>NOTICE</MenuItem>
|
||||||
<MenuItem value={6}>INFO</MenuItem>
|
<MenuItem value={6}>INFO</MenuItem>
|
||||||
<MenuItem value={7}>DEBUG</MenuItem>
|
|
||||||
<MenuItem value={9}>ALL</MenuItem>
|
<MenuItem value={9}>ALL</MenuItem>
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
@@ -330,6 +352,156 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
|
<Typography color="secondary">eMail</Typography>
|
||||||
|
<BlockFormControlLabel
|
||||||
|
control={
|
||||||
|
<Checkbox
|
||||||
|
checked={data.email_enabled}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
name="email_enabled"
|
||||||
|
disabled={!hardwareData.psram}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
label={
|
||||||
|
<Typography color={!hardwareData.psram ? 'grey' : 'default'}>
|
||||||
|
Enable eMail notification
|
||||||
|
{!hardwareData.psram && (
|
||||||
|
<Typography variant="caption">
|
||||||
|
({LL.IS_REQUIRED('PSRAM')})
|
||||||
|
</Typography>
|
||||||
|
)}
|
||||||
|
</Typography>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
{data.email_enabled && (
|
||||||
|
<>
|
||||||
|
<Grid
|
||||||
|
container
|
||||||
|
spacing={2}
|
||||||
|
direction="row"
|
||||||
|
justifyContent="flex-start"
|
||||||
|
alignItems="flex-start"
|
||||||
|
>
|
||||||
|
<Grid>
|
||||||
|
<ValidatedTextField
|
||||||
|
fieldErrors={fieldErrors || {}}
|
||||||
|
name="email_server"
|
||||||
|
label="SMTP Server"
|
||||||
|
variant="outlined"
|
||||||
|
value={data.email_server}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<ValidatedTextField
|
||||||
|
fieldErrors={fieldErrors || {}}
|
||||||
|
sx={{ width: '12ch' }}
|
||||||
|
name="email_port"
|
||||||
|
variant="outlined"
|
||||||
|
label="Port"
|
||||||
|
value={numberValue(data.email_port)}
|
||||||
|
type="number"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid size={4} mt={!data.email_ssl && !data.email_starttls ? 0 : 3}>
|
||||||
|
{!data.email_starttls && (
|
||||||
|
<BlockFormControlLabel
|
||||||
|
sx={{ width: '12ch' }}
|
||||||
|
control={
|
||||||
|
<Checkbox
|
||||||
|
checked={data.email_ssl}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
name="email_ssl"
|
||||||
|
disabled={
|
||||||
|
data.email_starttls || data.email_ssl === undefined
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
label="SSL/TLS"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{!data.email_ssl && (
|
||||||
|
<BlockFormControlLabel
|
||||||
|
sx={{ width: '12ch' }}
|
||||||
|
control={
|
||||||
|
<Checkbox
|
||||||
|
checked={data.email_starttls}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
name="email_starttls"
|
||||||
|
disabled={
|
||||||
|
data.email_ssl || data.email_starttls === undefined
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
label="STARTTLS"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
|
<Grid>
|
||||||
|
<ValidatedTextField
|
||||||
|
fieldErrors={fieldErrors || {}}
|
||||||
|
name="email_login"
|
||||||
|
label="Login"
|
||||||
|
variant="outlined"
|
||||||
|
value={data.email_login}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<ValidatedPasswordField
|
||||||
|
fieldErrors={fieldErrors || {}}
|
||||||
|
name="email_pass"
|
||||||
|
label="Password"
|
||||||
|
variant="outlined"
|
||||||
|
value={data.email_pass}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
|
<Grid>
|
||||||
|
<ValidatedTextField
|
||||||
|
fieldErrors={fieldErrors || {}}
|
||||||
|
name="email_sender"
|
||||||
|
label="From"
|
||||||
|
variant="outlined"
|
||||||
|
value={data.email_sender}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<ValidatedTextField
|
||||||
|
fieldErrors={fieldErrors || {}}
|
||||||
|
name="email_recp"
|
||||||
|
label="To"
|
||||||
|
variant="outlined"
|
||||||
|
value={data.email_recp}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<ValidatedTextField
|
||||||
|
fieldErrors={fieldErrors || {}}
|
||||||
|
name="email_subject"
|
||||||
|
label="Subject"
|
||||||
|
variant="outlined"
|
||||||
|
value={data.email_subject}
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
<Typography sx={{ pb: 1, pt: 2 }} variant="h6" color="primary">
|
<Typography sx={{ pb: 1, pt: 2 }} variant="h6" color="primary">
|
||||||
{LL.SENSORS()}
|
{LL.SENSORS()}
|
||||||
</Typography>
|
</Typography>
|
||||||
@@ -468,17 +640,25 @@ const ApplicationSettings = () => {
|
|||||||
name="board_profile"
|
name="board_profile"
|
||||||
label={LL.BOARD_PROFILE()}
|
label={LL.BOARD_PROFILE()}
|
||||||
value={data.board_profile}
|
value={data.board_profile}
|
||||||
disabled={processingBoard || hardwareData.model.startsWith('BBQKees')}
|
disabled={processingBoard}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onChange={changeBoardProfile}
|
onChange={changeBoardProfile}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
{boardProfileSelectItems()}
|
{hardwareData.model.startsWith('BBQKees') ? (
|
||||||
<Divider />
|
<MenuItem key={hardwareData.board} value={hardwareData.board}>
|
||||||
|
{BOARD_PROFILES[hardwareData.board as BoardProfileKey]}
|
||||||
|
</MenuItem>
|
||||||
|
) : (
|
||||||
|
boardProfileItems
|
||||||
|
)}
|
||||||
|
{(data.board_profile === 'CUSTOM' || data.developer_mode) && <Divider />}
|
||||||
|
{(data.board_profile === 'CUSTOM' || data.developer_mode) && (
|
||||||
<MenuItem key={'CUSTOM'} value={'CUSTOM'}>
|
<MenuItem key={'CUSTOM'} value={'CUSTOM'}>
|
||||||
{LL.CUSTOM()}…
|
{LL.CUSTOM()}…
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
|
)}
|
||||||
</TextField>
|
</TextField>
|
||||||
{data.board_profile === 'CUSTOM' && (
|
{data.board_profile === 'CUSTOM' && (
|
||||||
<>
|
<>
|
||||||
@@ -581,6 +761,7 @@ const ApplicationSettings = () => {
|
|||||||
<MenuItem value={0}>{LL.DISABLED(1)}</MenuItem>
|
<MenuItem value={0}>{LL.DISABLED(1)}</MenuItem>
|
||||||
<MenuItem value={1}>LAN8720</MenuItem>
|
<MenuItem value={1}>LAN8720</MenuItem>
|
||||||
<MenuItem value={2}>TLK110</MenuItem>
|
<MenuItem value={2}>TLK110</MenuItem>
|
||||||
|
<MenuItem value={3}>RTL8201</MenuItem>
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
@@ -836,8 +1017,9 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
{restartNeeded && (
|
{restartNeeded && (
|
||||||
<MessageBox my={2} level="warning" message={LL.RESTART_TEXT(0)}>
|
<MessageBox level="warning" message={LL.RESTART_TEXT(0)}>
|
||||||
<Button
|
<Button
|
||||||
|
sx={{ ml: 2 }}
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
variant="contained"
|
variant="contained"
|
||||||
color="error"
|
color="error"
|
||||||
@@ -873,10 +1055,12 @@ const ApplicationSettings = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return restarting ? (
|
||||||
|
<SystemMonitor />
|
||||||
|
) : (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{restarting ? <SystemMonitor /> : content()}
|
{content()}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import DownloadIcon from '@mui/icons-material/GetApp';
|
import DownloadIcon from '@mui/icons-material/GetApp';
|
||||||
@@ -19,6 +19,12 @@ import {
|
|||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
import { saveFile } from 'utils';
|
import { saveFile } from 'utils';
|
||||||
|
|
||||||
|
interface DownloadButton {
|
||||||
|
type: string;
|
||||||
|
label: string | number;
|
||||||
|
isGridButton: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
const DownloadUpload = () => {
|
const DownloadUpload = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
@@ -44,24 +50,78 @@ const DownloadUpload = () => {
|
|||||||
|
|
||||||
const { data, send: loadData, error } = useRequest(SystemApi.readSystemStatus);
|
const { data, send: loadData, error } = useRequest(SystemApi.readSystemStatus);
|
||||||
|
|
||||||
const doRestart = async () => {
|
const doRestart = useCallback(async () => {
|
||||||
setRestarting(true);
|
setRestarting(true);
|
||||||
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
try {
|
||||||
(error: Error) => {
|
await sendAPI({ device: 'system', cmd: 'restart', id: 0 });
|
||||||
toast.error(error.message);
|
} catch (error) {
|
||||||
|
toast.error((error as Error).message);
|
||||||
|
setRestarting(false);
|
||||||
}
|
}
|
||||||
);
|
}, [sendAPI]);
|
||||||
};
|
|
||||||
|
|
||||||
useLayoutTitle(LL.DOWNLOAD_UPLOAD());
|
useLayoutTitle(LL.DOWNLOAD_UPLOAD());
|
||||||
|
|
||||||
const content = () => {
|
const downloadButtons: DownloadButton[] = useMemo(
|
||||||
if (!data) {
|
() => [
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
{
|
||||||
|
type: 'settings',
|
||||||
|
label: LL.SETTINGS_OF(LL.APPLICATION()),
|
||||||
|
isGridButton: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'customizations',
|
||||||
|
label: LL.CUSTOMIZATIONS(),
|
||||||
|
isGridButton: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'entities',
|
||||||
|
label: LL.CUSTOM_ENTITIES(0),
|
||||||
|
isGridButton: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'schedule',
|
||||||
|
label: LL.SCHEDULE(0),
|
||||||
|
isGridButton: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'systembackup',
|
||||||
|
label: LL.DOWNLOAD_SYSTEM_BACKUP(),
|
||||||
|
isGridButton: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'allvalues',
|
||||||
|
label: LL.ALLVALUES(),
|
||||||
|
isGridButton: false
|
||||||
|
}
|
||||||
|
],
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleDownload = useCallback(
|
||||||
|
(type: string) => () => {
|
||||||
|
void sendExportData(type);
|
||||||
|
},
|
||||||
|
[sendExportData]
|
||||||
|
);
|
||||||
|
|
||||||
|
if (restarting) {
|
||||||
|
return <SystemMonitor />;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!data) {
|
||||||
return (
|
return (
|
||||||
<>
|
<SectionContent>
|
||||||
|
<FormLoader onRetry={loadData} errorMessage={error?.message || ''} />
|
||||||
|
</SectionContent>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const gridButtons = downloadButtons.filter((btn) => btn.isGridButton);
|
||||||
|
const standaloneButton = downloadButtons.find((btn) => !btn.isGridButton);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SectionContent>
|
||||||
<Typography sx={{ pb: 2 }} variant="h6" color="primary">
|
<Typography sx={{ pb: 2 }} variant="h6" color="primary">
|
||||||
{LL.DOWNLOAD(0)}
|
{LL.DOWNLOAD(0)}
|
||||||
</Typography>
|
</Typography>
|
||||||
@@ -69,54 +129,36 @@ const DownloadUpload = () => {
|
|||||||
<Typography mb={1} variant="body1" color="warning">
|
<Typography mb={1} variant="body1" color="warning">
|
||||||
{LL.DOWNLOAD_SETTINGS_TEXT()}.
|
{LL.DOWNLOAD_SETTINGS_TEXT()}.
|
||||||
</Typography>
|
</Typography>
|
||||||
<Grid container spacing={1}>
|
|
||||||
<Button
|
|
||||||
sx={{ ml: 2 }}
|
|
||||||
startIcon={<DownloadIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
color="primary"
|
|
||||||
onClick={() => sendExportData('settings')}
|
|
||||||
>
|
|
||||||
{LL.SETTINGS_OF(LL.APPLICATION())}
|
|
||||||
</Button>
|
|
||||||
|
|
||||||
|
<Grid container spacing={2}>
|
||||||
|
{gridButtons.map((button) => (
|
||||||
|
<Grid key={button.type}>
|
||||||
<Button
|
<Button
|
||||||
sx={{ ml: 2 }}
|
|
||||||
startIcon={<DownloadIcon />}
|
startIcon={<DownloadIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="primary"
|
color="primary"
|
||||||
onClick={() => sendExportData('customizations')}
|
onClick={handleDownload(button.type)}
|
||||||
>
|
>
|
||||||
{LL.CUSTOMIZATIONS()}
|
{button.label}
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
sx={{ ml: 2 }}
|
|
||||||
startIcon={<DownloadIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
color="primary"
|
|
||||||
onClick={() => sendExportData('entities')}
|
|
||||||
>
|
|
||||||
{LL.CUSTOM_ENTITIES(0)}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
sx={{ ml: 2 }}
|
|
||||||
startIcon={<DownloadIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
color="primary"
|
|
||||||
onClick={() => sendExportData('schedule')}
|
|
||||||
>
|
|
||||||
{LL.SCHEDULE(0)}
|
|
||||||
</Button>
|
</Button>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
))}
|
||||||
|
</Grid>
|
||||||
|
|
||||||
|
<Typography mt={2} mb={1} variant="body1" color="warning">
|
||||||
|
{LL.DOWNLOAD_SETTINGS_TEXT2()}.
|
||||||
|
</Typography>
|
||||||
|
|
||||||
|
{standaloneButton && (
|
||||||
<Button
|
<Button
|
||||||
sx={{ ml: 2, mt: 2 }}
|
|
||||||
startIcon={<DownloadIcon />}
|
startIcon={<DownloadIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="primary"
|
color="primary"
|
||||||
onClick={() => sendExportData('allvalues')}
|
onClick={handleDownload(standaloneButton.type)}
|
||||||
>
|
>
|
||||||
{LL.ALLVALUES()}
|
{standaloneButton.label}
|
||||||
</Button>
|
</Button>
|
||||||
|
)}
|
||||||
|
|
||||||
<Typography sx={{ pt: 2, pb: 2 }} variant="h6" color="primary">
|
<Typography sx={{ pt: 2, pb: 2 }} variant="h6" color="primary">
|
||||||
{LL.UPLOAD()}
|
{LL.UPLOAD()}
|
||||||
@@ -127,12 +169,7 @@ const DownloadUpload = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<SingleUpload text={LL.UPLOAD_DRAG()} doRestart={doRestart} />
|
<SingleUpload text={LL.UPLOAD_DRAG()} doRestart={doRestart} />
|
||||||
</>
|
</SectionContent>
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<SectionContent>{restarting ? <SystemMonitor /> : content()}</SectionContent>
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
import { useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
|
import SettingsBackupRestoreIcon from '@mui/icons-material/SettingsBackupRestore';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
import {
|
import {
|
||||||
|
Box,
|
||||||
Button,
|
Button,
|
||||||
Checkbox,
|
Checkbox,
|
||||||
Grid,
|
Grid,
|
||||||
@@ -30,6 +33,8 @@ import type { MqttSettingsType } from 'types';
|
|||||||
import { numberValue, updateValueDirty, useRest } from 'utils';
|
import { numberValue, updateValueDirty, useRest } from 'utils';
|
||||||
import { createMqttSettingsValidator, validate } from 'validators';
|
import { createMqttSettingsValidator, validate } from 'validators';
|
||||||
|
|
||||||
|
import { callAction } from '../../api/app';
|
||||||
|
|
||||||
const MqttSettings = () => {
|
const MqttSettings = () => {
|
||||||
const {
|
const {
|
||||||
loadData,
|
loadData,
|
||||||
@@ -52,23 +57,38 @@ const MqttSettings = () => {
|
|||||||
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
const sendResetMQTT = useCallback(() => {
|
||||||
origData,
|
void callAction({ action: 'resetMQTT' })
|
||||||
|
.then(() => {
|
||||||
|
toast.success('MQTT ' + LL.REFRESH() + ' successful');
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
toast.error(String(error.error?.message || 'An error occurred'));
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const updateFormValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValueDirty(
|
||||||
|
origData as unknown as Record<string, unknown>,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue as (value: unknown) => void
|
||||||
|
),
|
||||||
|
[origData, dirtyFlags, setDirtyFlags, updateDataValue]
|
||||||
);
|
);
|
||||||
|
|
||||||
const SecondsInputProps = {
|
const SecondsInputProps = useMemo(
|
||||||
|
() => ({
|
||||||
endAdornment: <InputAdornment position="end">{LL.SECONDS()}</InputAdornment>
|
endAdornment: <InputAdornment position="end">{LL.SECONDS()}</InputAdornment>
|
||||||
};
|
}),
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
const content = () => {
|
const emptyFieldErrors = useMemo(() => ({}), []);
|
||||||
if (!data) {
|
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
const validateAndSubmit = async () => {
|
const validateAndSubmit = useCallback(async () => {
|
||||||
|
if (!data) return;
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(createMqttSettingsValidator(data), data);
|
await validate(createMqttSettingsValidator(data), data);
|
||||||
@@ -76,10 +96,40 @@ const MqttSettings = () => {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
}, [data, saveData]);
|
||||||
|
|
||||||
|
const publishIntervalFields = useMemo(
|
||||||
|
() => [
|
||||||
|
{ name: 'publish_time_heartbeat', label: 'Heartbeat', validated: true },
|
||||||
|
{ name: 'publish_time_boiler', label: LL.MQTT_INT_BOILER(), validated: false },
|
||||||
|
{
|
||||||
|
name: 'publish_time_thermostat',
|
||||||
|
label: LL.MQTT_INT_THERMOSTATS(),
|
||||||
|
validated: false
|
||||||
|
},
|
||||||
|
{ name: 'publish_time_solar', label: LL.MQTT_INT_SOLAR(), validated: false },
|
||||||
|
{ name: 'publish_time_mixer', label: LL.MQTT_INT_MIXER(), validated: false },
|
||||||
|
{ name: 'publish_time_water', label: LL.MQTT_INT_WATER(), validated: false },
|
||||||
|
{ name: 'publish_time_sensor', label: LL.SENSORS(), validated: false },
|
||||||
|
{ name: 'publish_time_other', label: LL.DEFAULT(0), validated: false }
|
||||||
|
],
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!data) {
|
||||||
|
return (
|
||||||
|
<SectionContent>
|
||||||
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
|
<FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />
|
||||||
|
</SectionContent>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<SectionContent>
|
||||||
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
<>
|
<>
|
||||||
|
<Box display="flex" gap={2} mb={1}>
|
||||||
<BlockFormControlLabel
|
<BlockFormControlLabel
|
||||||
control={
|
control={
|
||||||
<Checkbox
|
<Checkbox
|
||||||
@@ -90,10 +140,21 @@ const MqttSettings = () => {
|
|||||||
}
|
}
|
||||||
label={LL.ENABLE_MQTT()}
|
label={LL.ENABLE_MQTT()}
|
||||||
/>
|
/>
|
||||||
|
{data.enabled && (
|
||||||
|
<Button
|
||||||
|
startIcon={<SettingsBackupRestoreIcon />}
|
||||||
|
color="secondary"
|
||||||
|
variant="outlined"
|
||||||
|
onClick={sendResetMQTT}
|
||||||
|
>
|
||||||
|
{LL.REFRESH() + ' MQTT'}
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</Box>
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
||||||
name="host"
|
name="host"
|
||||||
label={LL.ADDRESS_OF(LL.BROKER())}
|
label={LL.ADDRESS_OF(LL.BROKER())}
|
||||||
multiline
|
multiline
|
||||||
@@ -105,7 +166,7 @@ const MqttSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
||||||
name="port"
|
name="port"
|
||||||
label="Port"
|
label="Port"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -117,7 +178,7 @@ const MqttSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
||||||
name="base"
|
name="base"
|
||||||
label={LL.BASE_TOPIC()}
|
label={LL.BASE_TOPIC()}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -129,7 +190,7 @@ const MqttSettings = () => {
|
|||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<TextField
|
||||||
name="client_id"
|
name="client_id"
|
||||||
label={LL.ID_OF(LL.CLIENT()) + ' (' + LL.OPTIONAL() + ')'}
|
label={`${LL.ID_OF(LL.CLIENT())} (${LL.OPTIONAL()})`}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={data.client_id}
|
value={data.client_id}
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
@@ -158,7 +219,7 @@ const MqttSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
||||||
name="keep_alive"
|
name="keep_alive"
|
||||||
label="Keep Alive"
|
label="Keep Alive"
|
||||||
slotProps={{
|
slotProps={{
|
||||||
@@ -205,6 +266,7 @@ const MqttSettings = () => {
|
|||||||
label={LL.CERT()}
|
label={LL.CERT()}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={data.rootCA}
|
value={data.rootCA}
|
||||||
|
sx={{ width: '50ch' }}
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
@@ -283,7 +345,7 @@ const MqttSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
{/* <Grid container spacing={2} rowSpacing={0}> */}
|
||||||
<Grid>
|
<Grid>
|
||||||
<BlockFormControlLabel
|
<BlockFormControlLabel
|
||||||
control={
|
control={
|
||||||
@@ -336,127 +398,68 @@ const MqttSettings = () => {
|
|||||||
>
|
>
|
||||||
<MenuItem value={0}>{LL.MQTT_ENTITY_FORMAT_0()}</MenuItem>
|
<MenuItem value={0}>{LL.MQTT_ENTITY_FORMAT_0()}</MenuItem>
|
||||||
<MenuItem value={3}>
|
<MenuItem value={3}>
|
||||||
{LL.MQTT_ENTITY_FORMAT_1()} (v3.6)
|
{LL.MQTT_ENTITY_FORMAT_1()} (v3.5)
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
<MenuItem value={4}>
|
<MenuItem value={4}>
|
||||||
{LL.MQTT_ENTITY_FORMAT_2()} (v3.6)
|
{LL.MQTT_ENTITY_FORMAT_2()} (v3.5)
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={1}>
|
||||||
|
{LL.MQTT_ENTITY_FORMAT_1()} (latest)
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={2}>
|
||||||
|
{LL.MQTT_ENTITY_FORMAT_2()} (latest)
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
<MenuItem value={1}>{LL.MQTT_ENTITY_FORMAT_1()}</MenuItem>
|
|
||||||
<MenuItem value={2}>{LL.MQTT_ENTITY_FORMAT_2()}</MenuItem>
|
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
{data.discovery_type === 0 && (
|
||||||
|
<TextField
|
||||||
|
name="ha_number_mode"
|
||||||
|
label={LL.MQTT_INPUT_NUMBER_FORMAT()}
|
||||||
|
value={data.ha_number_mode}
|
||||||
|
variant="outlined"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
sx={{ width: '20ch' }}
|
||||||
|
margin="normal"
|
||||||
|
select
|
||||||
|
>
|
||||||
|
<MenuItem value={0}>Box</MenuItem>
|
||||||
|
<MenuItem value={1}>Slider</MenuItem>
|
||||||
|
</TextField>
|
||||||
|
)}
|
||||||
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
|
||||||
<Typography sx={{ pt: 2 }} variant="h6" color="primary">
|
<Typography sx={{ pt: 2 }} variant="h6" color="primary">
|
||||||
{LL.MQTT_PUBLISH_INTERVALS()} (0=auto)
|
{LL.MQTT_PUBLISH_INTERVALS()} (0=auto)
|
||||||
</Typography>
|
</Typography>
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid>
|
{publishIntervalFields.map((field) => (
|
||||||
|
<Grid key={field.name}>
|
||||||
|
{field.validated ? (
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
||||||
name="publish_time_heartbeat"
|
name={field.name}
|
||||||
label="Heartbeat"
|
label={field.label}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: SecondsInputProps
|
input: SecondsInputProps
|
||||||
}}
|
}}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={numberValue(data.publish_time_heartbeat)}
|
value={numberValue(
|
||||||
|
data[field.name as keyof MqttSettingsType] as number
|
||||||
|
)}
|
||||||
type="number"
|
type="number"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
</Grid>
|
) : (
|
||||||
<Grid>
|
|
||||||
<TextField
|
<TextField
|
||||||
name="publish_time_boiler"
|
name={field.name}
|
||||||
label={LL.MQTT_INT_BOILER()}
|
label={field.label}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={numberValue(data.publish_time_boiler)}
|
value={numberValue(
|
||||||
type="number"
|
data[field.name as keyof MqttSettingsType] as number
|
||||||
onChange={updateFormValue}
|
)}
|
||||||
margin="normal"
|
|
||||||
slotProps={{
|
|
||||||
input: SecondsInputProps
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<TextField
|
|
||||||
name="publish_time_thermostat"
|
|
||||||
label={LL.MQTT_INT_THERMOSTATS()}
|
|
||||||
variant="outlined"
|
|
||||||
value={numberValue(data.publish_time_thermostat)}
|
|
||||||
type="number"
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
slotProps={{
|
|
||||||
input: SecondsInputProps
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<TextField
|
|
||||||
name="publish_time_solar"
|
|
||||||
label={LL.MQTT_INT_SOLAR()}
|
|
||||||
variant="outlined"
|
|
||||||
value={numberValue(data.publish_time_solar)}
|
|
||||||
type="number"
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
slotProps={{
|
|
||||||
input: SecondsInputProps
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<TextField
|
|
||||||
name="publish_time_mixer"
|
|
||||||
label={LL.MQTT_INT_MIXER()}
|
|
||||||
variant="outlined"
|
|
||||||
value={numberValue(data.publish_time_mixer)}
|
|
||||||
type="number"
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
slotProps={{
|
|
||||||
input: SecondsInputProps
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<TextField
|
|
||||||
name="publish_time_water"
|
|
||||||
label={LL.MQTT_INT_WATER()}
|
|
||||||
variant="outlined"
|
|
||||||
value={numberValue(data.publish_time_water)}
|
|
||||||
type="number"
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
slotProps={{
|
|
||||||
input: SecondsInputProps
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<TextField
|
|
||||||
name="publish_time_sensor"
|
|
||||||
label={LL.SENSORS()}
|
|
||||||
variant="outlined"
|
|
||||||
value={numberValue(data.publish_time_sensor)}
|
|
||||||
type="number"
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
slotProps={{
|
|
||||||
input: SecondsInputProps
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<TextField
|
|
||||||
name="publish_time_other"
|
|
||||||
label={LL.DEFAULT(0)}
|
|
||||||
variant="outlined"
|
|
||||||
value={numberValue(data.publish_time_other)}
|
|
||||||
type="number"
|
type="number"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
@@ -464,7 +467,9 @@ const MqttSettings = () => {
|
|||||||
input: SecondsInputProps
|
input: SecondsInputProps
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
|
))}
|
||||||
</Grid>
|
</Grid>
|
||||||
{dirtyFlags && dirtyFlags.length !== 0 && (
|
{dirtyFlags && dirtyFlags.length !== 0 && (
|
||||||
<ButtonRow>
|
<ButtonRow>
|
||||||
@@ -491,13 +496,6 @@ const MqttSettings = () => {
|
|||||||
</ButtonRow>
|
</ButtonRow>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<SectionContent>
|
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
|
||||||
{content()}
|
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
||||||
@@ -39,7 +39,7 @@ import { formatLocalDateTime, updateValueDirty, useRest } from 'utils';
|
|||||||
import { validate } from 'validators';
|
import { validate } from 'validators';
|
||||||
import { NTP_SETTINGS_VALIDATOR } from 'validators/ntp';
|
import { NTP_SETTINGS_VALIDATOR } from 'validators/ntp';
|
||||||
|
|
||||||
import { TIME_ZONES, selectedTimeZone, timeZoneSelectItems } from './TZ';
|
import { TIME_ZONES, selectedTimeZone, useTimeZoneSelectItems } from './TZ';
|
||||||
|
|
||||||
const NTPSettings = () => {
|
const NTPSettings = () => {
|
||||||
const {
|
const {
|
||||||
@@ -61,9 +61,19 @@ const NTPSettings = () => {
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle('NTP');
|
useLayoutTitle('NTP');
|
||||||
|
|
||||||
|
// Memoized timezone select items for better performance
|
||||||
|
const timeZoneItems = useTimeZoneSelectItems();
|
||||||
|
|
||||||
|
// Memoized selected timezone value
|
||||||
|
const selectedTzValue = useMemo(
|
||||||
|
() => (data ? selectedTimeZone(data.tz_label, data.tz_format) : undefined),
|
||||||
|
[data?.tz_label, data?.tz_format]
|
||||||
|
);
|
||||||
|
|
||||||
const [localTime, setLocalTime] = useState<string>('');
|
const [localTime, setLocalTime] = useState<string>('');
|
||||||
const [settingTime, setSettingTime] = useState<boolean>(false);
|
const [settingTime, setSettingTime] = useState<boolean>(false);
|
||||||
const [processing, setProcessing] = useState<boolean>(false);
|
const [processing, setProcessing] = useState<boolean>(false);
|
||||||
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const { send: updateTime } = useRequest(
|
const { send: updateTime } = useRequest(
|
||||||
(local_time: Time) => NTPApi.updateTime(local_time),
|
(local_time: Time) => NTPApi.updateTime(local_time),
|
||||||
@@ -72,93 +82,52 @@ const NTPSettings = () => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
// Memoize updateFormValue to prevent recreation on every render
|
||||||
origData,
|
const updateFormValue = useMemo(
|
||||||
|
() =>
|
||||||
|
updateValueDirty(
|
||||||
|
origData as unknown as Record<string, unknown>,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue as (value: unknown) => void
|
||||||
|
),
|
||||||
|
[origData, dirtyFlags, setDirtyFlags, updateDataValue]
|
||||||
);
|
);
|
||||||
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
// Memoize updateLocalTime handler
|
||||||
|
const updateLocalTime = useCallback(
|
||||||
|
(event: React.ChangeEvent<HTMLInputElement>) => setLocalTime(event.target.value),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
const updateLocalTime = (event: React.ChangeEvent<HTMLInputElement>) =>
|
// Memoize openSetTime handler
|
||||||
setLocalTime(event.target.value);
|
const openSetTime = useCallback(() => {
|
||||||
|
|
||||||
const openSetTime = () => {
|
|
||||||
setLocalTime(formatLocalDateTime(new Date()));
|
setLocalTime(formatLocalDateTime(new Date()));
|
||||||
setSettingTime(true);
|
setSettingTime(true);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const configureTime = async () => {
|
// Memoize configureTime handler
|
||||||
|
const configureTime = useCallback(async () => {
|
||||||
setProcessing(true);
|
setProcessing(true);
|
||||||
|
|
||||||
await updateTime({ local_time: formatLocalDateTime(new Date(localTime)) })
|
try {
|
||||||
.then(async () => {
|
await updateTime({ local_time: formatLocalDateTime(new Date(localTime)) });
|
||||||
toast.success(LL.TIME_SET());
|
toast.success(LL.TIME_SET());
|
||||||
setSettingTime(false);
|
setSettingTime(false);
|
||||||
await loadData();
|
await loadData();
|
||||||
})
|
} catch {
|
||||||
.catch(() => {
|
|
||||||
toast.error(LL.PROBLEM_UPDATING());
|
toast.error(LL.PROBLEM_UPDATING());
|
||||||
})
|
} finally {
|
||||||
.finally(() => {
|
|
||||||
setProcessing(false);
|
setProcessing(false);
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const renderSetTimeDialog = () => (
|
|
||||||
<Dialog
|
|
||||||
sx={dialogStyle}
|
|
||||||
open={settingTime}
|
|
||||||
onClose={() => setSettingTime(false)}
|
|
||||||
>
|
|
||||||
<DialogTitle>{LL.SET_TIME(1)}</DialogTitle>
|
|
||||||
<DialogContent dividers>
|
|
||||||
<Box color="warning.main" p={0} pl={0} pr={0} mt={0} mb={2}>
|
|
||||||
<Typography variant="body2">{LL.SET_TIME_TEXT()}</Typography>
|
|
||||||
</Box>
|
|
||||||
<TextField
|
|
||||||
label={LL.LOCAL_TIME(0)}
|
|
||||||
type="datetime-local"
|
|
||||||
value={localTime}
|
|
||||||
onChange={updateLocalTime}
|
|
||||||
disabled={processing}
|
|
||||||
fullWidth
|
|
||||||
slotProps={{
|
|
||||||
inputLabel: {
|
|
||||||
shrink: true
|
|
||||||
}
|
}
|
||||||
}}
|
}, [localTime, updateTime, LL, loadData]);
|
||||||
/>
|
|
||||||
</DialogContent>
|
|
||||||
<DialogActions>
|
|
||||||
<Button
|
|
||||||
startIcon={<CancelIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
onClick={() => setSettingTime(false)}
|
|
||||||
color="secondary"
|
|
||||||
>
|
|
||||||
{LL.CANCEL()}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
startIcon={<AccessTimeIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
onClick={configureTime}
|
|
||||||
disabled={processing}
|
|
||||||
color="primary"
|
|
||||||
>
|
|
||||||
{LL.UPDATE()}
|
|
||||||
</Button>
|
|
||||||
</DialogActions>
|
|
||||||
</Dialog>
|
|
||||||
);
|
|
||||||
|
|
||||||
const content = () => {
|
// Memoize close dialog handler
|
||||||
if (!data) {
|
const handleCloseSetTime = useCallback(() => setSettingTime(false), []);
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
const validateAndSubmit = async () => {
|
// Memoize validate and submit handler
|
||||||
|
const validateAndSubmit = useCallback(async () => {
|
||||||
|
if (!data) return;
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(NTP_SETTINGS_VALIDATOR, data);
|
await validate(NTP_SETTINGS_VALIDATOR, data);
|
||||||
@@ -166,16 +135,26 @@ const NTPSettings = () => {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
}, [data, saveData]);
|
||||||
|
|
||||||
const changeTimeZone = (event: React.ChangeEvent<HTMLInputElement>) => {
|
// Memoize timezone change handler
|
||||||
|
const changeTimeZone = useCallback(
|
||||||
|
(event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
void updateState(readNTPSettings(), (settings: NTPSettingsType) => ({
|
void updateState(readNTPSettings(), (settings: NTPSettingsType) => ({
|
||||||
...settings,
|
...settings,
|
||||||
tz_label: event.target.value,
|
tz_label: event.target.value,
|
||||||
tz_format: TIME_ZONES[event.target.value]
|
tz_format: TIME_ZONES[event.target.value]
|
||||||
}));
|
}));
|
||||||
updateFormValue(event);
|
updateFormValue(event);
|
||||||
};
|
},
|
||||||
|
[updateFormValue]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Memoize render content to prevent unnecessary re-renders
|
||||||
|
const renderContent = useMemo(() => {
|
||||||
|
if (!data) {
|
||||||
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -205,13 +184,13 @@ const NTPSettings = () => {
|
|||||||
label={LL.TIME_ZONE()}
|
label={LL.TIME_ZONE()}
|
||||||
fullWidth
|
fullWidth
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={selectedTimeZone(data.tz_label, data.tz_format)}
|
value={selectedTzValue}
|
||||||
onChange={changeTimeZone}
|
onChange={changeTimeZone}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
<MenuItem disabled>{LL.TIME_ZONE()}...</MenuItem>
|
<MenuItem disabled>{LL.TIME_ZONE()}...</MenuItem>
|
||||||
{timeZoneSelectItems()}
|
{timeZoneItems}
|
||||||
</ValidatedTextField>
|
</ValidatedTextField>
|
||||||
|
|
||||||
<Box display="flex" flexWrap="wrap">
|
<Box display="flex" flexWrap="wrap">
|
||||||
@@ -230,7 +209,6 @@ const NTPSettings = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
{renderSetTimeDialog()}
|
|
||||||
|
|
||||||
{dirtyFlags && dirtyFlags.length !== 0 && (
|
{dirtyFlags && dirtyFlags.length !== 0 && (
|
||||||
<ButtonRow>
|
<ButtonRow>
|
||||||
@@ -258,12 +236,66 @@ const NTPSettings = () => {
|
|||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
}, [
|
||||||
|
data,
|
||||||
|
errorMessage,
|
||||||
|
loadData,
|
||||||
|
updateFormValue,
|
||||||
|
fieldErrors,
|
||||||
|
selectedTzValue,
|
||||||
|
changeTimeZone,
|
||||||
|
timeZoneItems,
|
||||||
|
dirtyFlags,
|
||||||
|
openSetTime,
|
||||||
|
saving,
|
||||||
|
validateAndSubmit,
|
||||||
|
LL
|
||||||
|
]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{content()}
|
{renderContent}
|
||||||
|
<Dialog sx={dialogStyle} open={settingTime} onClose={handleCloseSetTime}>
|
||||||
|
<DialogTitle>{LL.SET_TIME(1)}</DialogTitle>
|
||||||
|
<DialogContent dividers>
|
||||||
|
<Box color="warning.main" p={0} pl={0} pr={0} mt={0} mb={2}>
|
||||||
|
<Typography variant="body2">{LL.SET_TIME_TEXT()}</Typography>
|
||||||
|
</Box>
|
||||||
|
<TextField
|
||||||
|
label={LL.LOCAL_TIME(0)}
|
||||||
|
type="datetime-local"
|
||||||
|
value={localTime}
|
||||||
|
onChange={updateLocalTime}
|
||||||
|
disabled={processing}
|
||||||
|
fullWidth
|
||||||
|
slotProps={{
|
||||||
|
inputLabel: {
|
||||||
|
shrink: true
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</DialogContent>
|
||||||
|
<DialogActions>
|
||||||
|
<Button
|
||||||
|
startIcon={<CancelIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
onClick={handleCloseSetTime}
|
||||||
|
color="secondary"
|
||||||
|
>
|
||||||
|
{LL.CANCEL()}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
startIcon={<AccessTimeIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
onClick={configureTime}
|
||||||
|
disabled={processing}
|
||||||
|
color="primary"
|
||||||
|
>
|
||||||
|
{LL.UPDATE()}
|
||||||
|
</Button>
|
||||||
|
</DialogActions>
|
||||||
|
</Dialog>
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -17,6 +17,7 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
|
Divider,
|
||||||
List
|
List
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
|
|
||||||
@@ -29,54 +30,38 @@ import { SectionContent, useLayoutTitle } from 'components';
|
|||||||
import ListMenuItem from 'components/layout/ListMenuItem';
|
import ListMenuItem from 'components/layout/ListMenuItem';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
|
|
||||||
|
import SystemMonitor from '../status/SystemMonitor';
|
||||||
|
|
||||||
const Settings = () => {
|
const Settings = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle(LL.SETTINGS(0));
|
useLayoutTitle(LL.SETTINGS(0));
|
||||||
|
|
||||||
const [confirmFactoryReset, setConfirmFactoryReset] = useState<boolean>(false);
|
const [confirmFactoryReset, setConfirmFactoryReset] = useState(false);
|
||||||
|
const [restarting, setRestarting] = useState<boolean>();
|
||||||
|
|
||||||
const { send: sendAPI } = useRequest((data: APIcall) => API(data), {
|
const { send: sendAPI } = useRequest((data: APIcall) => API(data), {
|
||||||
immediate: false
|
immediate: false
|
||||||
});
|
});
|
||||||
|
|
||||||
const doFormat = async () => {
|
const doFormat = useCallback(async () => {
|
||||||
await sendAPI({ device: 'system', cmd: 'format', id: 0 }).then(() => {
|
await sendAPI({ device: 'system', cmd: 'format', id: 0 }).then(() => {
|
||||||
|
setRestarting(true);
|
||||||
setConfirmFactoryReset(false);
|
setConfirmFactoryReset(false);
|
||||||
});
|
});
|
||||||
};
|
}, [sendAPI]);
|
||||||
|
|
||||||
const renderFactoryResetDialog = () => (
|
const handleFactoryResetClose = useCallback(() => {
|
||||||
<Dialog
|
setConfirmFactoryReset(false);
|
||||||
sx={dialogStyle}
|
}, []);
|
||||||
open={confirmFactoryReset}
|
|
||||||
onClose={() => setConfirmFactoryReset(false)}
|
|
||||||
>
|
|
||||||
<DialogTitle>{LL.FACTORY_RESET()}</DialogTitle>
|
|
||||||
<DialogContent dividers>{LL.SYSTEM_FACTORY_TEXT_DIALOG()}</DialogContent>
|
|
||||||
<DialogActions>
|
|
||||||
<Button
|
|
||||||
startIcon={<CancelIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
onClick={() => setConfirmFactoryReset(false)}
|
|
||||||
color="secondary"
|
|
||||||
>
|
|
||||||
{LL.CANCEL()}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
startIcon={<SettingsBackupRestoreIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
onClick={doFormat}
|
|
||||||
color="error"
|
|
||||||
>
|
|
||||||
{LL.FACTORY_RESET()}
|
|
||||||
</Button>
|
|
||||||
</DialogActions>
|
|
||||||
</Dialog>
|
|
||||||
);
|
|
||||||
|
|
||||||
const content = () => (
|
const handleFactoryResetClick = useCallback(() => {
|
||||||
|
setConfirmFactoryReset(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const content = useMemo(() => {
|
||||||
|
return (
|
||||||
<>
|
<>
|
||||||
<List sx={{ borderRadius: 3, border: '2px solid grey' }}>
|
<List>
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
icon={TuneIcon}
|
icon={TuneIcon}
|
||||||
bgcolor="#134ba2"
|
bgcolor="#134ba2"
|
||||||
@@ -141,13 +126,46 @@ const Settings = () => {
|
|||||||
/>
|
/>
|
||||||
</List>
|
</List>
|
||||||
|
|
||||||
{renderFactoryResetDialog()}
|
<Dialog
|
||||||
|
sx={dialogStyle}
|
||||||
<Box mt={2} display="flex" flexWrap="wrap">
|
open={confirmFactoryReset}
|
||||||
|
onClose={handleFactoryResetClose}
|
||||||
|
>
|
||||||
|
<DialogTitle>{LL.FACTORY_RESET()}</DialogTitle>
|
||||||
|
<DialogContent dividers>{LL.SYSTEM_FACTORY_TEXT_DIALOG()}</DialogContent>
|
||||||
|
<DialogActions>
|
||||||
|
<Button
|
||||||
|
startIcon={<CancelIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
onClick={handleFactoryResetClose}
|
||||||
|
color="secondary"
|
||||||
|
>
|
||||||
|
{LL.CANCEL()}
|
||||||
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<SettingsBackupRestoreIcon />}
|
startIcon={<SettingsBackupRestoreIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={() => setConfirmFactoryReset(true)}
|
onClick={doFormat}
|
||||||
|
color="error"
|
||||||
|
>
|
||||||
|
{LL.FACTORY_RESET()}
|
||||||
|
</Button>
|
||||||
|
</DialogActions>
|
||||||
|
</Dialog>
|
||||||
|
|
||||||
|
<Divider />
|
||||||
|
|
||||||
|
<Box
|
||||||
|
mt={2}
|
||||||
|
display="flex"
|
||||||
|
justifyContent="flex-end"
|
||||||
|
flexWrap="nowrap"
|
||||||
|
whiteSpace="nowrap"
|
||||||
|
>
|
||||||
|
<Button
|
||||||
|
startIcon={<SettingsBackupRestoreIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
onClick={handleFactoryResetClick}
|
||||||
color="error"
|
color="error"
|
||||||
>
|
>
|
||||||
{LL.FACTORY_RESET()}
|
{LL.FACTORY_RESET()}
|
||||||
@@ -155,8 +173,16 @@ const Settings = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
}, [
|
||||||
|
LL,
|
||||||
|
handleFactoryResetClick,
|
||||||
|
handleFactoryResetClose,
|
||||||
|
doFormat,
|
||||||
|
confirmFactoryReset,
|
||||||
|
restarting
|
||||||
|
]);
|
||||||
|
|
||||||
return <SectionContent>{content()}</SectionContent>;
|
return restarting ? <SystemMonitor /> : <SectionContent>{content}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Settings;
|
export default Settings;
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
|
import { useMemo } from 'react';
|
||||||
|
|
||||||
import { MenuItem } from '@mui/material';
|
import { MenuItem } from '@mui/material';
|
||||||
|
|
||||||
type TimeZones = Record<string, string>;
|
export const TIME_ZONES: Record<string, string> = {
|
||||||
|
|
||||||
export const TIME_ZONES: TimeZones = {
|
|
||||||
'Africa/Abidjan': 'GMT0',
|
'Africa/Abidjan': 'GMT0',
|
||||||
'Africa/Accra': 'GMT0',
|
'Africa/Accra': 'GMT0',
|
||||||
'Africa/Addis_Ababa': 'EAT-3',
|
'Africa/Addis_Ababa': 'EAT-3',
|
||||||
@@ -465,14 +465,33 @@ export const TIME_ZONES: TimeZones = {
|
|||||||
'Pacific/Wallis': 'UNK-12'
|
'Pacific/Wallis': 'UNK-12'
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Pre-compute sorted timezone labels for better performance
|
||||||
|
export const TIME_ZONE_LABELS = Object.keys(TIME_ZONES).sort();
|
||||||
|
|
||||||
export function selectedTimeZone(label: string, format: string) {
|
export function selectedTimeZone(label: string, format: string) {
|
||||||
return TIME_ZONES[label] === format ? label : undefined;
|
return TIME_ZONES[label] === format ? label : undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function timeZoneSelectItems() {
|
// Memoized version for use in components
|
||||||
return Object.keys(TIME_ZONES).map((label) => (
|
export function useTimeZoneSelectItems() {
|
||||||
|
return useMemo(
|
||||||
|
() =>
|
||||||
|
TIME_ZONE_LABELS.map((label) => (
|
||||||
|
<MenuItem key={label} value={label}>
|
||||||
|
{label}
|
||||||
|
</MenuItem>
|
||||||
|
)),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback export for backward compatibility - now memoized
|
||||||
|
const precomputedTimeZoneItems = TIME_ZONE_LABELS.map((label) => (
|
||||||
<MenuItem key={label} value={label}>
|
<MenuItem key={label} value={label}>
|
||||||
{label}
|
{label}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
));
|
));
|
||||||
|
|
||||||
|
export function timeZoneSelectItems() {
|
||||||
|
return precomputedTimeZoneItems;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useState } from 'react';
|
import { memo, useCallback, useMemo, useState } from 'react';
|
||||||
import {
|
import {
|
||||||
Navigate,
|
Navigate,
|
||||||
Route,
|
Route,
|
||||||
@@ -28,8 +28,7 @@ const Network = () => {
|
|||||||
[
|
[
|
||||||
{
|
{
|
||||||
path: '/settings/network/settings',
|
path: '/settings/network/settings',
|
||||||
element: <NetworkSettings />,
|
element: <NetworkSettings />
|
||||||
dog: 'woof'
|
|
||||||
},
|
},
|
||||||
{ path: '/settings/network/scan', element: <WiFiNetworkScanner /> }
|
{ path: '/settings/network/scan', element: <WiFiNetworkScanner /> }
|
||||||
],
|
],
|
||||||
@@ -53,14 +52,17 @@ const Network = () => {
|
|||||||
setSelectedNetwork(undefined);
|
setSelectedNetwork(undefined);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
const contextValue = useMemo(
|
||||||
<WiFiConnectionContext.Provider
|
() => ({
|
||||||
value={{
|
|
||||||
...(selectedNetwork && { selectedNetwork }),
|
...(selectedNetwork && { selectedNetwork }),
|
||||||
selectNetwork,
|
selectNetwork,
|
||||||
deselectNetwork
|
deselectNetwork
|
||||||
}}
|
}),
|
||||||
>
|
[selectedNetwork, selectNetwork, deselectNetwork]
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WiFiConnectionContext.Provider value={contextValue}>
|
||||||
<RouterTabs value={routerTab}>
|
<RouterTabs value={routerTab}>
|
||||||
<Tab
|
<Tab
|
||||||
value="/settings/network/settings"
|
value="/settings/network/settings"
|
||||||
@@ -80,4 +82,4 @@ const Network = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Network;
|
export default memo(Network);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useContext, useEffect, useState } from 'react';
|
import { memo, useCallback, useContext, useEffect, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -109,14 +109,8 @@ const NetworkSettings = () => {
|
|||||||
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
useEffect(() => deselectNetwork, [deselectNetwork]);
|
const validateAndSubmit = useCallback(async () => {
|
||||||
|
if (!data) return;
|
||||||
const content = () => {
|
|
||||||
if (!data) {
|
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
const validateAndSubmit = async () => {
|
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(createNetworkSettingsValidator(data), data);
|
await validate(createNetworkSettingsValidator(data), data);
|
||||||
@@ -125,21 +119,26 @@ const NetworkSettings = () => {
|
|||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
deselectNetwork();
|
deselectNetwork();
|
||||||
};
|
}, [data, saveData, deselectNetwork]);
|
||||||
|
|
||||||
const setCancel = async () => {
|
const setCancel = useCallback(async () => {
|
||||||
deselectNetwork();
|
deselectNetwork();
|
||||||
await loadData();
|
await loadData();
|
||||||
};
|
}, [deselectNetwork, loadData]);
|
||||||
|
|
||||||
const doRestart = async () => {
|
const doRestart = useCallback(async () => {
|
||||||
setRestarting(true);
|
setRestarting(true);
|
||||||
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
||||||
(error: Error) => {
|
(error: Error) => {
|
||||||
toast.error(error.message);
|
toast.error(error.message);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
};
|
}, [sendAPI]);
|
||||||
|
|
||||||
|
const content = () => {
|
||||||
|
if (!data) {
|
||||||
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -165,7 +164,7 @@ const NetworkSettings = () => {
|
|||||||
selectedNetwork.bssid
|
selectedNetwork.bssid
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<IconButton onClick={setCancel}>
|
<IconButton onClick={setCancel} aria-label={LL.CANCEL()}>
|
||||||
<DeleteIcon />
|
<DeleteIcon />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
@@ -356,8 +355,9 @@ const NetworkSettings = () => {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{restartNeeded && (
|
{restartNeeded && (
|
||||||
<MessageBox my={2} level="warning" message={LL.RESTART_TEXT(0)}>
|
<MessageBox level="warning" message={LL.RESTART_TEXT(0)}>
|
||||||
<Button
|
<Button
|
||||||
|
sx={{ ml: 2 }}
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
variant="contained"
|
variant="contained"
|
||||||
color="error"
|
color="error"
|
||||||
@@ -397,12 +397,14 @@ const NetworkSettings = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return restarting ? (
|
||||||
|
<SystemMonitor />
|
||||||
|
) : (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{restarting ? <SystemMonitor /> : content()}
|
{content()}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default NetworkSettings;
|
export default memo(NetworkSettings);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useRef, useState } from 'react';
|
import { memo, useCallback, useRef, useState } from 'react';
|
||||||
|
|
||||||
import PermScanWifiIcon from '@mui/icons-material/PermScanWifi';
|
import PermScanWifiIcon from '@mui/icons-material/PermScanWifi';
|
||||||
import { Button } from '@mui/material';
|
import { Button } from '@mui/material';
|
||||||
@@ -48,12 +48,12 @@ const WiFiNetworkScanner = () => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const renderNetworkScanner = () => {
|
const renderNetworkScanner = useCallback(() => {
|
||||||
if (!networkList) {
|
if (!networkList) {
|
||||||
return <FormLoader errorMessage={errorMessage || ''} />;
|
return <FormLoader errorMessage={errorMessage || ''} />;
|
||||||
}
|
}
|
||||||
return <WiFiNetworkSelector networkList={networkList} />;
|
return <WiFiNetworkSelector networkList={networkList} />;
|
||||||
};
|
}, [networkList, errorMessage]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
@@ -73,4 +73,4 @@ const WiFiNetworkScanner = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default WiFiNetworkScanner;
|
export default memo(WiFiNetworkScanner);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useContext } from 'react';
|
import { memo, useCallback, useContext } from 'react';
|
||||||
|
|
||||||
import LockIcon from '@mui/icons-material/Lock';
|
import LockIcon from '@mui/icons-material/Lock';
|
||||||
import LockOpenIcon from '@mui/icons-material/LockOpen';
|
import LockOpenIcon from '@mui/icons-material/LockOpen';
|
||||||
@@ -63,7 +63,8 @@ const WiFiNetworkSelector = ({ networkList }: { networkList: WiFiNetworkList })
|
|||||||
|
|
||||||
const wifiConnectionContext = useContext(WiFiConnectionContext);
|
const wifiConnectionContext = useContext(WiFiConnectionContext);
|
||||||
|
|
||||||
const renderNetwork = (network: WiFiNetwork) => (
|
const renderNetwork = useCallback(
|
||||||
|
(network: WiFiNetwork) => (
|
||||||
<ListItem
|
<ListItem
|
||||||
key={network.bssid}
|
key={network.bssid}
|
||||||
onClick={() => wifiConnectionContext.selectNetwork(network)}
|
onClick={() => wifiConnectionContext.selectNetwork(network)}
|
||||||
@@ -88,13 +89,15 @@ const WiFiNetworkSelector = ({ networkList }: { networkList: WiFiNetworkList })
|
|||||||
</Badge>
|
</Badge>
|
||||||
</ListItemIcon>
|
</ListItemIcon>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
),
|
||||||
|
[wifiConnectionContext, theme]
|
||||||
);
|
);
|
||||||
|
|
||||||
if (networkList.networks.length === 0) {
|
if (networkList.networks.length === 0) {
|
||||||
return <MessageBox mt={2} mb={1} message={LL.NETWORK_NO_WIFI()} level="info" />;
|
return <MessageBox message={LL.NETWORK_NO_WIFI()} level="info" />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return <List>{networkList.networks.map(renderNetwork)}</List>;
|
return <List>{networkList.networks.map(renderNetwork)}</List>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default WiFiNetworkSelector;
|
export default memo(WiFiNetworkSelector);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useEffect } from 'react';
|
import { memo, useEffect } from 'react';
|
||||||
|
|
||||||
import CloseIcon from '@mui/icons-material/Close';
|
import CloseIcon from '@mui/icons-material/Close';
|
||||||
import {
|
import {
|
||||||
@@ -40,7 +40,7 @@ const GenerateToken = ({ username, onClose }: GenerateTokenProps) => {
|
|||||||
if (open) {
|
if (open) {
|
||||||
void generateToken();
|
void generateToken();
|
||||||
}
|
}
|
||||||
}, [open]);
|
}, [open, generateToken]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
@@ -86,4 +86,4 @@ const GenerateToken = ({ username, onClose }: GenerateTokenProps) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default GenerateToken;
|
export default memo(GenerateToken);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useContext, useState } from 'react';
|
import { memo, useCallback, useContext, useMemo, useState } from 'react';
|
||||||
import { useBlocker } from 'react-router';
|
import { useBlocker } from 'react-router';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -55,7 +55,9 @@ const ManageUsers = () => {
|
|||||||
const blocker = useBlocker(changed !== 0);
|
const blocker = useBlocker(changed !== 0);
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
const table_theme = useTheme({
|
const table_theme = useMemo(
|
||||||
|
() =>
|
||||||
|
useTheme({
|
||||||
Table: `
|
Table: `
|
||||||
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) minmax(120px, max-content) 120px;
|
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) minmax(120px, max-content) 120px;
|
||||||
`,
|
`,
|
||||||
@@ -93,41 +95,45 @@ const ManageUsers = () => {
|
|||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
});
|
}),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
const content = () => {
|
const noAdminConfigured = useCallback(
|
||||||
if (!data) {
|
() => !data?.users.find((u) => u.admin),
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
[data]
|
||||||
}
|
);
|
||||||
|
|
||||||
const noAdminConfigured = () => !data.users.find((u) => u.admin);
|
const removeUser = useCallback(
|
||||||
|
(toRemove: UserType) => {
|
||||||
const removeUser = (toRemove: UserType) => {
|
if (!data) return;
|
||||||
const users = data.users.filter((u) => u.username !== toRemove.username);
|
const users = data.users.filter((u) => u.username !== toRemove.username);
|
||||||
updateDataValue({ ...data, users });
|
updateDataValue({ ...data, users });
|
||||||
setChanged(changed + 1);
|
setChanged(changed + 1);
|
||||||
};
|
},
|
||||||
|
[data, updateDataValue, changed]
|
||||||
|
);
|
||||||
|
|
||||||
const createUser = () => {
|
const createUser = useCallback(() => {
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
setUser({
|
setUser({
|
||||||
username: '',
|
username: '',
|
||||||
password: '',
|
password: '',
|
||||||
admin: true
|
admin: true
|
||||||
});
|
});
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const editUser = (toEdit: UserType) => {
|
const editUser = useCallback((toEdit: UserType) => {
|
||||||
setCreating(false);
|
setCreating(false);
|
||||||
setUser({ ...toEdit });
|
setUser({ ...toEdit });
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const cancelEditingUser = () => {
|
const cancelEditingUser = useCallback(() => {
|
||||||
setUser(undefined);
|
setUser(undefined);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const doneEditingUser = () => {
|
const doneEditingUser = useCallback(() => {
|
||||||
if (user) {
|
if (user && data) {
|
||||||
const users = [
|
const users = [
|
||||||
...data.users.filter(
|
...data.users.filter(
|
||||||
(u: { username: string }) => u.username !== user.username
|
(u: { username: string }) => u.username !== user.username
|
||||||
@@ -138,26 +144,31 @@ const ManageUsers = () => {
|
|||||||
setUser(undefined);
|
setUser(undefined);
|
||||||
setChanged(changed + 1);
|
setChanged(changed + 1);
|
||||||
}
|
}
|
||||||
};
|
}, [user, data, updateDataValue, changed]);
|
||||||
|
|
||||||
const closeGenerateToken = () => {
|
const closeGenerateToken = useCallback(() => {
|
||||||
setGeneratingToken(undefined);
|
setGeneratingToken(undefined);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const generateToken = (username: string) => {
|
const generateTokenForUser = useCallback((username: string) => {
|
||||||
setGeneratingToken(username);
|
setGeneratingToken(username);
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const onSubmit = async () => {
|
const onSubmit = useCallback(async () => {
|
||||||
await saveData();
|
await saveData();
|
||||||
await authenticatedContext.refresh();
|
await authenticatedContext.refresh();
|
||||||
setChanged(0);
|
setChanged(0);
|
||||||
};
|
}, [saveData, authenticatedContext]);
|
||||||
|
|
||||||
const onCancelSubmit = async () => {
|
const onCancelSubmit = useCallback(async () => {
|
||||||
await loadData();
|
await loadData();
|
||||||
setChanged(0);
|
setChanged(0);
|
||||||
};
|
}, [loadData]);
|
||||||
|
|
||||||
|
const content = () => {
|
||||||
|
if (!data) {
|
||||||
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
||||||
|
}
|
||||||
|
|
||||||
interface UserType2 {
|
interface UserType2 {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -167,10 +178,14 @@ const ManageUsers = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// add id to the type, needed for the table
|
// add id to the type, needed for the table
|
||||||
const user_table = data.users.map((u) => ({
|
const user_table = useMemo(
|
||||||
|
() =>
|
||||||
|
data.users.map((u) => ({
|
||||||
...u,
|
...u,
|
||||||
id: u.username
|
id: u.username
|
||||||
})) as UserType2[];
|
})) as UserType2[],
|
||||||
|
[data.users]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -196,15 +211,24 @@ const ManageUsers = () => {
|
|||||||
<Cell stiff>
|
<Cell stiff>
|
||||||
<IconButton
|
<IconButton
|
||||||
size="small"
|
size="small"
|
||||||
|
aria-label={LL.GENERATING_TOKEN()}
|
||||||
disabled={!authenticatedContext.me.admin}
|
disabled={!authenticatedContext.me.admin}
|
||||||
onClick={() => generateToken(u.username)}
|
onClick={() => generateTokenForUser(u.username)}
|
||||||
>
|
>
|
||||||
<VpnKeyIcon />
|
<VpnKeyIcon />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
<IconButton size="small" onClick={() => removeUser(u)}>
|
<IconButton
|
||||||
|
size="small"
|
||||||
|
onClick={() => removeUser(u)}
|
||||||
|
aria-label={LL.REMOVE()}
|
||||||
|
>
|
||||||
<DeleteIcon />
|
<DeleteIcon />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
<IconButton size="small" onClick={() => editUser(u)}>
|
<IconButton
|
||||||
|
size="small"
|
||||||
|
onClick={() => editUser(u)}
|
||||||
|
aria-label={LL.EDIT()}
|
||||||
|
>
|
||||||
<EditIcon />
|
<EditIcon />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</Cell>
|
</Cell>
|
||||||
@@ -286,4 +310,4 @@ const ManageUsers = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default ManageUsers;
|
export default memo(ManageUsers);
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { memo, useMemo } from 'react';
|
||||||
import { Navigate, Route, Routes, matchRoutes, useLocation } from 'react-router';
|
import { Navigate, Route, Routes, matchRoutes, useLocation } from 'react-router';
|
||||||
|
|
||||||
import { Tab } from '@mui/material';
|
import { Tab } from '@mui/material';
|
||||||
@@ -12,12 +13,21 @@ const Security = () => {
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle(LL.SECURITY(0));
|
useLayoutTitle(LL.SECURITY(0));
|
||||||
|
|
||||||
const matchedRoutes = matchRoutes(
|
const location = useLocation();
|
||||||
|
|
||||||
|
const matchedRoutes = useMemo(
|
||||||
|
() =>
|
||||||
|
matchRoutes(
|
||||||
[
|
[
|
||||||
{ path: '/settings/security/settings', element: <ManageUsers />, dog: 'woof' },
|
{
|
||||||
|
path: '/settings/security/settings',
|
||||||
|
element: <ManageUsers />
|
||||||
|
},
|
||||||
{ path: '/settings/security/users', element: <SecuritySettings /> }
|
{ path: '/settings/security/users', element: <SecuritySettings /> }
|
||||||
],
|
],
|
||||||
useLocation()
|
location
|
||||||
|
),
|
||||||
|
[location]
|
||||||
);
|
);
|
||||||
const routerTab = matchedRoutes?.[0]?.route.path || false;
|
const routerTab = matchedRoutes?.[0]?.route.path || false;
|
||||||
|
|
||||||
@@ -42,4 +52,4 @@ const Security = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Security;
|
export default memo(Security);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useContext, useState } from 'react';
|
import { memo, useCallback, useContext, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
@@ -44,18 +44,14 @@ const SecuritySettings = () => {
|
|||||||
const authenticatedContext = useContext(AuthenticatedContext);
|
const authenticatedContext = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
const updateFormValue = updateValueDirty(
|
||||||
origData,
|
origData as unknown as Record<string, unknown>,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue as (value: unknown) => void
|
||||||
);
|
);
|
||||||
|
|
||||||
const content = () => {
|
const validateAndSubmit = useCallback(async () => {
|
||||||
if (!data) {
|
if (!data) return;
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
const validateAndSubmit = async () => {
|
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(SECURITY_SETTINGS_VALIDATOR, data);
|
await validate(SECURITY_SETTINGS_VALIDATOR, data);
|
||||||
@@ -64,7 +60,12 @@ const SecuritySettings = () => {
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
};
|
}, [data, saveData, authenticatedContext]);
|
||||||
|
|
||||||
|
const content = () => {
|
||||||
|
if (!data) {
|
||||||
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -115,4 +116,4 @@ const SecuritySettings = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default SecuritySettings;
|
export default memo(SecuritySettings);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { memo, useCallback, useEffect, useState } from 'react';
|
||||||
import type { FC } from 'react';
|
import type { FC } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -45,7 +45,14 @@ const User: FC<UserFormProps> = ({
|
|||||||
}) => {
|
}) => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
const updateFormValue = updateValue(setUser);
|
const updateFormValue = updateValue((updater) => {
|
||||||
|
setUser((prevState) => {
|
||||||
|
if (!prevState) return prevState;
|
||||||
|
return updater(
|
||||||
|
prevState as unknown as Record<string, unknown>
|
||||||
|
) as unknown as UserType;
|
||||||
|
});
|
||||||
|
});
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
const open = !!user;
|
const open = !!user;
|
||||||
|
|
||||||
@@ -55,7 +62,7 @@ const User: FC<UserFormProps> = ({
|
|||||||
}
|
}
|
||||||
}, [open]);
|
}, [open]);
|
||||||
|
|
||||||
const validateAndDone = async () => {
|
const validateAndDone = useCallback(async () => {
|
||||||
if (user) {
|
if (user) {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
@@ -65,7 +72,7 @@ const User: FC<UserFormProps> = ({
|
|||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
}, [user, validator, onDoneEditing]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
@@ -137,4 +144,4 @@ const User: FC<UserFormProps> = ({
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default User;
|
export default memo(User);
|
||||||
|
|||||||
@@ -34,19 +34,10 @@ export const apStatusHighlight = ({ status }: APStatusType, theme: Theme) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const APStatus = () => {
|
const getApStatusText = (
|
||||||
const { data, send: loadData, error } = useRequest(APApi.readAPStatus);
|
status: APNetworkStatus,
|
||||||
|
LL: ReturnType<typeof useI18nContext>['LL']
|
||||||
useInterval(() => {
|
) => {
|
||||||
void loadData();
|
|
||||||
});
|
|
||||||
|
|
||||||
const { LL } = useI18nContext();
|
|
||||||
useLayoutTitle(LL.ACCESS_POINT(0));
|
|
||||||
|
|
||||||
const theme = useTheme();
|
|
||||||
|
|
||||||
const apStatus = ({ status }: APStatusType) => {
|
|
||||||
switch (status) {
|
switch (status) {
|
||||||
case APNetworkStatus.ACTIVE:
|
case APNetworkStatus.ACTIVE:
|
||||||
return LL.ACTIVE();
|
return LL.ACTIVE();
|
||||||
@@ -59,12 +50,27 @@ const APStatus = () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const content = () => {
|
const APStatus = () => {
|
||||||
|
const { data, send: loadData, error } = useRequest(APApi.readAPStatus);
|
||||||
|
const { LL } = useI18nContext();
|
||||||
|
const theme = useTheme();
|
||||||
|
|
||||||
|
useLayoutTitle(LL.ACCESS_POINT(0));
|
||||||
|
|
||||||
|
useInterval(() => {
|
||||||
|
void loadData();
|
||||||
|
});
|
||||||
|
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return (
|
||||||
|
<SectionContent>
|
||||||
|
<FormLoader onRetry={loadData} errorMessage={error?.message || ''} />
|
||||||
|
</SectionContent>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<SectionContent>
|
||||||
<List>
|
<List>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
@@ -72,19 +78,26 @@ const APStatus = () => {
|
|||||||
<SettingsInputAntennaIcon />
|
<SettingsInputAntennaIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary={LL.STATUS_OF('')} secondary={apStatus(data)} />
|
<ListItemText
|
||||||
|
primary={LL.STATUS_OF('')}
|
||||||
|
secondary={getApStatusText(data.status, LL)}
|
||||||
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
|
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar>IP</Avatar>
|
<Avatar sx={{ bgcolor: 'primary.main' }}>IP</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary={LL.ADDRESS_OF('IP')} secondary={data.ip_address} />
|
<ListItemText primary={LL.ADDRESS_OF('IP')} secondary={data.ip_address} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
|
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar>
|
<Avatar sx={{ bgcolor: 'primary.main' }}>
|
||||||
<DeviceHubIcon />
|
<DeviceHubIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
@@ -93,21 +106,22 @@ const APStatus = () => {
|
|||||||
secondary={data.mac_address}
|
secondary={data.mac_address}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
|
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar>
|
<Avatar sx={{ bgcolor: 'primary.main' }}>
|
||||||
<ComputerIcon />
|
<ComputerIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary={LL.AP_CLIENTS()} secondary={data.station_num} />
|
<ListItemText primary={LL.AP_CLIENTS()} secondary={data.station_num} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
</List>
|
</List>
|
||||||
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return <SectionContent>{content()}</SectionContent>;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default APStatus;
|
export default APStatus;
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { useCallback, useMemo } from 'react';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Body,
|
Body,
|
||||||
Cell,
|
Cell,
|
||||||
@@ -17,6 +19,12 @@ import { useInterval } from 'utils';
|
|||||||
import { readActivity } from '../../api/app';
|
import { readActivity } from '../../api/app';
|
||||||
import type { Stat } from '../main/types';
|
import type { Stat } from '../main/types';
|
||||||
|
|
||||||
|
const QUALITY_COLORS = {
|
||||||
|
PERFECT: '#00FF7F',
|
||||||
|
WARNING: 'orange',
|
||||||
|
POOR: 'red'
|
||||||
|
} as const;
|
||||||
|
|
||||||
const SystemActivity = () => {
|
const SystemActivity = () => {
|
||||||
const { data, send: loadData, error } = useRequest(readActivity);
|
const { data, send: loadData, error } = useRequest(readActivity);
|
||||||
|
|
||||||
@@ -28,7 +36,9 @@ const SystemActivity = () => {
|
|||||||
|
|
||||||
useLayoutTitle(LL.DATA_TRAFFIC());
|
useLayoutTitle(LL.DATA_TRAFFIC());
|
||||||
|
|
||||||
const stats_theme = tableTheme({
|
const stats_theme = tableTheme(
|
||||||
|
useMemo(
|
||||||
|
() => ({
|
||||||
Table: `
|
Table: `
|
||||||
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 90px 90px 80px;
|
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 90px 90px 80px;
|
||||||
`,
|
`,
|
||||||
@@ -64,29 +74,35 @@ const SystemActivity = () => {
|
|||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
});
|
}),
|
||||||
|
[]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
const showName = (id: number) => {
|
const showName = useCallback(
|
||||||
|
(id: number) => {
|
||||||
const name: keyof Translation['STATUS_NAMES'] =
|
const name: keyof Translation['STATUS_NAMES'] =
|
||||||
id.toString() as keyof Translation['STATUS_NAMES'];
|
id.toString() as keyof Translation['STATUS_NAMES'];
|
||||||
return LL.STATUS_NAMES[name]();
|
return LL.STATUS_NAMES[name]();
|
||||||
};
|
},
|
||||||
|
[LL]
|
||||||
|
);
|
||||||
|
|
||||||
const showQuality = (stat: Stat) => {
|
const showQuality = useCallback((stat: Stat) => {
|
||||||
if (stat.q === 0 || stat.s + stat.f === 0) {
|
if (stat.q === 0 || stat.s + stat.f === 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (stat.q === 100) {
|
if (stat.q === 100) {
|
||||||
return <div style={{ color: '#00FF7F' }}>{stat.q}%</div>;
|
return <div style={{ color: QUALITY_COLORS.PERFECT }}>{stat.q}%</div>;
|
||||||
}
|
}
|
||||||
if (stat.q >= 95) {
|
if (stat.q >= 95) {
|
||||||
return <div style={{ color: 'orange' }}>{stat.q}%</div>;
|
return <div style={{ color: QUALITY_COLORS.WARNING }}>{stat.q}%</div>;
|
||||||
} else {
|
} else {
|
||||||
return <div style={{ color: 'red' }}>{stat.q}%</div>;
|
return <div style={{ color: QUALITY_COLORS.POOR }}>{stat.q}%</div>;
|
||||||
}
|
}
|
||||||
};
|
}, []);
|
||||||
|
|
||||||
const content = () => {
|
const content = useMemo(() => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
||||||
}
|
}
|
||||||
@@ -121,9 +137,9 @@ const SystemActivity = () => {
|
|||||||
)}
|
)}
|
||||||
</Table>
|
</Table>
|
||||||
);
|
);
|
||||||
};
|
}, [data, loadData, error?.message, stats_theme, LL, showName, showQuality]);
|
||||||
|
|
||||||
return <SectionContent>{content()}</SectionContent>;
|
return <SectionContent>{content}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default SystemActivity;
|
export default SystemActivity;
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { ReactElement } from 'react';
|
||||||
|
|
||||||
import AppsIcon from '@mui/icons-material/Apps';
|
import AppsIcon from '@mui/icons-material/Apps';
|
||||||
import DeveloperBoardIcon from '@mui/icons-material/DeveloperBoard';
|
import DeveloperBoardIcon from '@mui/icons-material/DeveloperBoard';
|
||||||
import DevicesIcon from '@mui/icons-material/Devices';
|
import DevicesIcon from '@mui/icons-material/Devices';
|
||||||
@@ -24,10 +26,61 @@ import { useInterval } from 'utils';
|
|||||||
|
|
||||||
import BBQKeesIcon from './bbqkees.svg';
|
import BBQKeesIcon from './bbqkees.svg';
|
||||||
|
|
||||||
|
// Constants
|
||||||
|
const AVATAR_COLORS = {
|
||||||
|
DEFAULT: '#5f9a5f',
|
||||||
|
BBQKEES: '#003289'
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
const TEMP_THRESHOLD_CELSIUS = 90; // Temperature threshold to determine F vs C
|
||||||
|
|
||||||
function formatNumber(num: number) {
|
function formatNumber(num: number) {
|
||||||
return new Intl.NumberFormat().format(num);
|
return new Intl.NumberFormat().format(num);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function formatTemperature(temp?: number): string {
|
||||||
|
if (!temp) return '';
|
||||||
|
const unit = temp > TEMP_THRESHOLD_CELSIUS ? 'F' : 'C';
|
||||||
|
return `, T: ${temp} °${unit}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatFlashSpeed(speed: number): string {
|
||||||
|
return (speed / 1000000).toFixed(0) + ' MHz';
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatCPUCores(cores: number): string {
|
||||||
|
return cores === 1 ? 'single-core)' : 'dual-core)';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reusable component for hardware status list items
|
||||||
|
interface HardwareListItemProps {
|
||||||
|
icon: ReactElement;
|
||||||
|
primary: string;
|
||||||
|
secondary: string;
|
||||||
|
avatarColor?: string;
|
||||||
|
customIcon?: ReactElement | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const HardwareListItem = ({
|
||||||
|
icon,
|
||||||
|
primary,
|
||||||
|
secondary,
|
||||||
|
avatarColor = AVATAR_COLORS.DEFAULT,
|
||||||
|
customIcon
|
||||||
|
}: HardwareListItemProps) => (
|
||||||
|
<>
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: avatarColor, color: 'white' }}>
|
||||||
|
{customIcon || icon}
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText primary={primary} secondary={secondary} />
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
const HardwareStatus = () => {
|
const HardwareStatus = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
@@ -39,175 +92,72 @@ const HardwareStatus = () => {
|
|||||||
void loadData();
|
void loadData();
|
||||||
});
|
});
|
||||||
|
|
||||||
const content = () => {
|
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return (
|
||||||
|
<SectionContent>
|
||||||
|
<FormLoader onRetry={loadData} errorMessage={error?.message || ''} />
|
||||||
|
</SectionContent>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<SectionContent>
|
||||||
<List>
|
<List>
|
||||||
<ListItem>
|
<HardwareListItem
|
||||||
<ListItemAvatar>
|
icon={<TapAndPlayIcon />}
|
||||||
{data.model ? (
|
primary={`${LL.HARDWARE()} ${LL.DEVICE()}`}
|
||||||
<Avatar sx={{ bgcolor: '#003289', color: 'white' }}>
|
secondary={data.model || data.cpu_type}
|
||||||
|
avatarColor={data.model ? AVATAR_COLORS.BBQKEES : AVATAR_COLORS.DEFAULT}
|
||||||
|
customIcon={
|
||||||
|
data.model ? (
|
||||||
<img
|
<img
|
||||||
alt="BBQKees"
|
alt="BBQKees"
|
||||||
src={BBQKeesIcon}
|
src={BBQKeesIcon}
|
||||||
style={{ width: 16, verticalAlign: 'middle' }}
|
style={{ width: 16, verticalAlign: 'middle' }}
|
||||||
/>
|
/>
|
||||||
</Avatar>
|
) : undefined
|
||||||
) : (
|
}
|
||||||
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
|
||||||
<TapAndPlayIcon />
|
|
||||||
</Avatar>
|
|
||||||
)}
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary={LL.HARDWARE() + ' ' + LL.DEVICE()}
|
|
||||||
secondary={data.model ? data.model : data.cpu_type}
|
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
<HardwareListItem
|
||||||
<Divider variant="inset" component="li" />
|
icon={<DevicesIcon />}
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
|
||||||
<DevicesIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary="SDK"
|
primary="SDK"
|
||||||
secondary={data.arduino_version + ' / ESP-IDF ' + data.sdk_version}
|
secondary={`${data.arduino_version} / ESP-IDF ${data.sdk_version}`}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
<HardwareListItem
|
||||||
<Divider variant="inset" component="li" />
|
icon={<DeveloperBoardIcon />}
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
|
||||||
<DeveloperBoardIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary="CPU"
|
primary="CPU"
|
||||||
secondary={
|
secondary={`${data.esp_platform}/${data.cpu_type} (rev.${data.cpu_rev}, ${formatCPUCores(data.cpu_cores)} @ ${data.cpu_freq_mhz} Mhz${formatTemperature(data.temperature)}`}
|
||||||
data.esp_platform +
|
|
||||||
'/' +
|
|
||||||
data.cpu_type +
|
|
||||||
' (rev.' +
|
|
||||||
data.cpu_rev +
|
|
||||||
', ' +
|
|
||||||
(data.cpu_cores === 1 ? 'single-core)' : 'dual-core)') +
|
|
||||||
' @ ' +
|
|
||||||
data.cpu_freq_mhz +
|
|
||||||
' Mhz' +
|
|
||||||
// bit of a hack : if the CPU temp is higher than 90 (=32 Fahrenheit if using Celsius), show F, otherwise C
|
|
||||||
(data.temperature
|
|
||||||
? ', T: ' +
|
|
||||||
data.temperature +
|
|
||||||
' °' +
|
|
||||||
(data.temperature > 90 ? 'F' : 'C')
|
|
||||||
: '')
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
<HardwareListItem
|
||||||
<Divider variant="inset" component="li" />
|
icon={<MemoryIcon />}
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
|
||||||
<MemoryIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary={LL.FREE_MEMORY()}
|
primary={LL.FREE_MEMORY()}
|
||||||
secondary={
|
secondary={`${formatNumber(data.free_heap)} KB (${formatNumber(data.max_alloc_heap)} KB max alloc, ${formatNumber(data.free_caps)} KB caps)`}
|
||||||
formatNumber(data.free_heap) +
|
|
||||||
' KB (' +
|
|
||||||
formatNumber(data.max_alloc_heap) +
|
|
||||||
' KB max alloc, ' +
|
|
||||||
formatNumber(data.free_caps) +
|
|
||||||
' KB caps)'
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
|
||||||
{data.psram_size !== undefined && data.free_psram !== undefined && (
|
{data.psram_size !== undefined && data.free_psram !== undefined && (
|
||||||
<>
|
<HardwareListItem
|
||||||
<Divider variant="inset" component="li" />
|
icon={<AppsIcon />}
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
|
||||||
<AppsIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary={LL.PSRAM()}
|
primary={LL.PSRAM()}
|
||||||
secondary={
|
secondary={`${formatNumber(data.psram_size)} KB / ${formatNumber(data.free_psram)} KB`}
|
||||||
formatNumber(data.psram_size) +
|
|
||||||
' KB / ' +
|
|
||||||
formatNumber(data.free_psram) +
|
|
||||||
' KB'
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
|
||||||
</>
|
|
||||||
)}
|
)}
|
||||||
<Divider variant="inset" component="li" />
|
<HardwareListItem
|
||||||
<ListItem>
|
icon={<SdStorageIcon />}
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
|
||||||
<SdStorageIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary={LL.FLASH()}
|
primary={LL.FLASH()}
|
||||||
secondary={
|
secondary={`${formatNumber(data.flash_chip_size)} KB , ${formatFlashSpeed(data.flash_chip_speed)}`}
|
||||||
formatNumber(data.flash_chip_size) +
|
|
||||||
' KB , ' +
|
|
||||||
(data.flash_chip_speed / 1000000).toFixed(0) +
|
|
||||||
' MHz'
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
<HardwareListItem
|
||||||
<Divider variant="inset" component="li" />
|
icon={<SdCardAlertIcon />}
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
|
||||||
<SdCardAlertIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary={LL.APPSIZE()}
|
primary={LL.APPSIZE()}
|
||||||
secondary={
|
secondary={`${data.partition}: ${formatNumber(data.app_used)} KB / ${formatNumber(data.app_free)} KB`}
|
||||||
data.partition +
|
|
||||||
': ' +
|
|
||||||
formatNumber(data.app_used) +
|
|
||||||
' KB / ' +
|
|
||||||
formatNumber(data.app_free) +
|
|
||||||
' KB'
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
<HardwareListItem
|
||||||
<Divider variant="inset" component="li" />
|
icon={<FolderIcon />}
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
|
||||||
<FolderIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary={LL.FILESYSTEM()}
|
primary={LL.FILESYSTEM()}
|
||||||
secondary={
|
secondary={`${formatNumber(data.fs_used)} KB / ${formatNumber(data.fs_free)} KB`}
|
||||||
formatNumber(data.fs_used) +
|
|
||||||
' KB / ' +
|
|
||||||
formatNumber(data.fs_free) +
|
|
||||||
' KB'
|
|
||||||
}
|
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
|
||||||
<Divider variant="inset" component="li" />
|
|
||||||
</List>
|
</List>
|
||||||
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return <SectionContent>{content()}</SectionContent>;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default HardwareStatus;
|
export default HardwareStatus;
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { type FC, memo, useMemo } from 'react';
|
||||||
|
|
||||||
import AutoAwesomeMotionIcon from '@mui/icons-material/AutoAwesomeMotion';
|
import AutoAwesomeMotionIcon from '@mui/icons-material/AutoAwesomeMotion';
|
||||||
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
||||||
import ReportIcon from '@mui/icons-material/Report';
|
import ReportIcon from '@mui/icons-material/Report';
|
||||||
@@ -22,17 +24,28 @@ import type { MqttStatusType } from 'types';
|
|||||||
import { MqttDisconnectReason } from 'types';
|
import { MqttDisconnectReason } from 'types';
|
||||||
import { useInterval } from 'utils';
|
import { useInterval } from 'utils';
|
||||||
|
|
||||||
|
// Disconnect reason lookup table - created once, reused across renders
|
||||||
|
const DISCONNECT_REASONS: Record<MqttDisconnectReason, string> = {
|
||||||
|
[MqttDisconnectReason.USER_OK]: 'User disconnected',
|
||||||
|
[MqttDisconnectReason.TCP_DISCONNECTED]: 'TCP disconnected',
|
||||||
|
[MqttDisconnectReason.MQTT_UNACCEPTABLE_PROTOCOL_VERSION]:
|
||||||
|
'Unacceptable protocol version',
|
||||||
|
[MqttDisconnectReason.MQTT_IDENTIFIER_REJECTED]: 'Client ID rejected',
|
||||||
|
[MqttDisconnectReason.MQTT_SERVER_UNAVAILABLE]: 'Server unavailable',
|
||||||
|
[MqttDisconnectReason.MQTT_MALFORMED_CREDENTIALS]: 'Malformed credentials',
|
||||||
|
[MqttDisconnectReason.MQTT_NOT_AUTHORIZED]: 'Not authorized',
|
||||||
|
[MqttDisconnectReason.TLS_BAD_FINGERPRINT]: 'TLS fingerprint invalid'
|
||||||
|
};
|
||||||
|
|
||||||
|
const getDisconnectReason = (disconnect_reason: MqttDisconnectReason): string =>
|
||||||
|
DISCONNECT_REASONS[disconnect_reason] ?? 'Unknown';
|
||||||
|
|
||||||
export const mqttStatusHighlight = (
|
export const mqttStatusHighlight = (
|
||||||
{ enabled, connected }: MqttStatusType,
|
{ enabled, connected }: MqttStatusType,
|
||||||
theme: Theme
|
theme: Theme
|
||||||
) => {
|
) => {
|
||||||
if (!enabled) {
|
if (!enabled) return theme.palette.info.main;
|
||||||
return theme.palette.info.main;
|
return connected ? theme.palette.success.main : theme.palette.error.main;
|
||||||
}
|
|
||||||
if (connected) {
|
|
||||||
return theme.palette.success.main;
|
|
||||||
}
|
|
||||||
return theme.palette.error.main;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const mqttPublishHighlight = (
|
export const mqttPublishHighlight = (
|
||||||
@@ -41,68 +54,22 @@ export const mqttPublishHighlight = (
|
|||||||
) => {
|
) => {
|
||||||
if (mqtt_fails === 0) return theme.palette.success.main;
|
if (mqtt_fails === 0) return theme.palette.success.main;
|
||||||
if (mqtt_fails < 10) return theme.palette.warning.main;
|
if (mqtt_fails < 10) return theme.palette.warning.main;
|
||||||
|
|
||||||
return theme.palette.error.main;
|
return theme.palette.error.main;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const mqttQueueHighlight = (
|
export const mqttQueueHighlight = ({ mqtt_queued }: MqttStatusType, theme: Theme) =>
|
||||||
{ mqtt_queued }: MqttStatusType,
|
mqtt_queued <= 1 ? theme.palette.success.main : theme.palette.warning.main;
|
||||||
theme: Theme
|
|
||||||
) => {
|
|
||||||
if (mqtt_queued <= 1) return theme.palette.success.main;
|
|
||||||
|
|
||||||
return theme.palette.warning.main;
|
interface ConnectionStatusProps {
|
||||||
};
|
data: MqttStatusType;
|
||||||
|
theme: Theme;
|
||||||
const MqttStatus = () => {
|
}
|
||||||
const { data, send: loadData, error } = useRequest(MqttApi.readMqttStatus);
|
|
||||||
|
|
||||||
useInterval(() => {
|
|
||||||
void loadData();
|
|
||||||
});
|
|
||||||
|
|
||||||
|
// Memoized component to prevent unnecessary re-renders when parent updates
|
||||||
|
const ConnectionStatus: FC<ConnectionStatusProps> = memo(({ data, theme }) => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle('MQTT');
|
|
||||||
|
|
||||||
const theme = useTheme();
|
return (
|
||||||
|
|
||||||
const mqttStatus = ({ enabled, connected, connect_count }: MqttStatusType) => {
|
|
||||||
if (!enabled) {
|
|
||||||
return LL.NOT_ENABLED();
|
|
||||||
}
|
|
||||||
if (connected) {
|
|
||||||
return LL.CONNECTED(0) + ' (' + connect_count + ')';
|
|
||||||
}
|
|
||||||
return LL.DISCONNECTED() + ' (' + connect_count + ')';
|
|
||||||
};
|
|
||||||
|
|
||||||
const disconnectReason = ({ disconnect_reason }: MqttStatusType) => {
|
|
||||||
switch (disconnect_reason) {
|
|
||||||
case MqttDisconnectReason.TCP_DISCONNECTED:
|
|
||||||
return 'TCP disconnected';
|
|
||||||
case MqttDisconnectReason.MQTT_UNACCEPTABLE_PROTOCOL_VERSION:
|
|
||||||
return 'Unacceptable protocol version';
|
|
||||||
case MqttDisconnectReason.MQTT_IDENTIFIER_REJECTED:
|
|
||||||
return 'Client ID rejected';
|
|
||||||
case MqttDisconnectReason.MQTT_SERVER_UNAVAILABLE:
|
|
||||||
return 'Server unavailable';
|
|
||||||
case MqttDisconnectReason.MQTT_MALFORMED_CREDENTIALS:
|
|
||||||
return 'Malformed credentials';
|
|
||||||
case MqttDisconnectReason.MQTT_NOT_AUTHORIZED:
|
|
||||||
return 'Not authorized';
|
|
||||||
case MqttDisconnectReason.TLS_BAD_FINGERPRINT:
|
|
||||||
return 'TLS fingerprint invalid';
|
|
||||||
default:
|
|
||||||
return 'Unknown';
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const content = () => {
|
|
||||||
if (!data) {
|
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
const renderConnectionStatus = () => (
|
|
||||||
<>
|
<>
|
||||||
{!data.connected && (
|
{!data.connected && (
|
||||||
<>
|
<>
|
||||||
@@ -114,7 +81,7 @@ const MqttStatus = () => {
|
|||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
primary={LL.DISCONNECT_REASON()}
|
primary={LL.DISCONNECT_REASON()}
|
||||||
secondary={disconnectReason(data)}
|
secondary={getDisconnectReason(data.disconnect_reason)}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
@@ -147,8 +114,40 @@ const MqttStatus = () => {
|
|||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
const MqttStatus = () => {
|
||||||
|
const { data, send: loadData, error } = useRequest(MqttApi.readMqttStatus);
|
||||||
|
const { LL } = useI18nContext();
|
||||||
|
const theme = useTheme();
|
||||||
|
|
||||||
|
useLayoutTitle('MQTT');
|
||||||
|
|
||||||
|
useInterval(() => {
|
||||||
|
void loadData();
|
||||||
|
});
|
||||||
|
|
||||||
|
// Memoize error message separately to avoid re-renders on error object changes
|
||||||
|
const errorMessage = error?.message || '';
|
||||||
|
|
||||||
|
const mqttStatusText = useMemo(() => {
|
||||||
|
if (!data) return '';
|
||||||
|
if (!data.enabled) return LL.NOT_ENABLED();
|
||||||
|
return data.connected
|
||||||
|
? `${LL.CONNECTED(0)} (${data.connect_count})`
|
||||||
|
: `${LL.DISCONNECTED()} (${data.connect_count})`;
|
||||||
|
}, [data, LL]);
|
||||||
|
|
||||||
|
if (!data) {
|
||||||
|
return (
|
||||||
|
<SectionContent>
|
||||||
|
<FormLoader onRetry={loadData} errorMessage={errorMessage} />
|
||||||
|
</SectionContent>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<SectionContent>
|
||||||
<List>
|
<List>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
@@ -156,15 +155,13 @@ const MqttStatus = () => {
|
|||||||
<DeviceHubIcon />
|
<DeviceHubIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary={LL.STATUS_OF('')} secondary={mqttStatus(data)} />
|
<ListItemText primary={LL.STATUS_OF('')} secondary={mqttStatusText} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
{data.enabled && renderConnectionStatus()}
|
{data.enabled && <ConnectionStatus data={data} theme={theme} />}
|
||||||
</List>
|
</List>
|
||||||
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return <SectionContent>{content()}</SectionContent>;
|
|
||||||
};
|
|
||||||
|
|
||||||
export default MqttStatus;
|
export default MqttStatus;
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { useMemo } from 'react';
|
||||||
|
|
||||||
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
||||||
import DnsIcon from '@mui/icons-material/Dns';
|
import DnsIcon from '@mui/icons-material/Dns';
|
||||||
import SwapVerticalCircleIcon from '@mui/icons-material/SwapVerticalCircle';
|
import SwapVerticalCircleIcon from '@mui/icons-material/SwapVerticalCircle';
|
||||||
@@ -23,18 +25,7 @@ import { NTPSyncStatus } from 'types';
|
|||||||
import { useInterval } from 'utils';
|
import { useInterval } from 'utils';
|
||||||
import { formatDateTime } from 'utils';
|
import { formatDateTime } from 'utils';
|
||||||
|
|
||||||
const NTPStatus = () => {
|
// Utility functions
|
||||||
const { data, send: loadData, error } = useRequest(NTPApi.readNTPStatus);
|
|
||||||
|
|
||||||
useInterval(() => {
|
|
||||||
void loadData();
|
|
||||||
});
|
|
||||||
|
|
||||||
const { LL } = useI18nContext();
|
|
||||||
useLayoutTitle('NTP');
|
|
||||||
|
|
||||||
NTPApi.updateTime;
|
|
||||||
|
|
||||||
const isNtpEnabled = ({ status }: NTPStatusType) =>
|
const isNtpEnabled = ({ status }: NTPStatusType) =>
|
||||||
status !== NTPSyncStatus.NTP_DISABLED;
|
status !== NTPSyncStatus.NTP_DISABLED;
|
||||||
|
|
||||||
@@ -51,6 +42,16 @@ const NTPStatus = () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const NTPStatus = () => {
|
||||||
|
const { data, send: loadData, error } = useRequest(NTPApi.readNTPStatus);
|
||||||
|
|
||||||
|
useInterval(() => {
|
||||||
|
void loadData();
|
||||||
|
});
|
||||||
|
|
||||||
|
const { LL } = useI18nContext();
|
||||||
|
useLayoutTitle('NTP');
|
||||||
|
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
const ntpStatus = ({ status }: NTPStatusType) => {
|
const ntpStatus = ({ status }: NTPStatusType) => {
|
||||||
@@ -66,13 +67,12 @@ const NTPStatus = () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const content = () => {
|
const content = useMemo(() => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
|
||||||
<List>
|
<List>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
@@ -121,11 +121,10 @@ const NTPStatus = () => {
|
|||||||
</ListItem>
|
</ListItem>
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
</List>
|
</List>
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
};
|
}, [data, error, loadData, LL, theme]);
|
||||||
|
|
||||||
return <SectionContent>{content()}</SectionContent>;
|
return <SectionContent>{content}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default NTPStatus;
|
export default NTPStatus;
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
import { useMemo } from 'react';
|
||||||
|
|
||||||
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
||||||
import DnsIcon from '@mui/icons-material/Dns';
|
import DnsIcon from '@mui/icons-material/Dns';
|
||||||
import GiteIcon from '@mui/icons-material/Gite';
|
import GiteIcon from '@mui/icons-material/Gite';
|
||||||
@@ -25,10 +27,17 @@ import type { NetworkStatusType } from 'types';
|
|||||||
import { NetworkConnectionStatus } from 'types';
|
import { NetworkConnectionStatus } from 'types';
|
||||||
import { useInterval } from 'utils';
|
import { useInterval } from 'utils';
|
||||||
|
|
||||||
|
// Utility functions
|
||||||
const isConnected = ({ status }: NetworkStatusType) =>
|
const isConnected = ({ status }: NetworkStatusType) =>
|
||||||
status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED ||
|
status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED ||
|
||||||
status === NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED;
|
status === NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED;
|
||||||
|
|
||||||
|
export const isWiFi = ({ status }: NetworkStatusType) =>
|
||||||
|
status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED;
|
||||||
|
|
||||||
|
export const isEthernet = ({ status }: NetworkStatusType) =>
|
||||||
|
status === NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED;
|
||||||
|
|
||||||
const networkStatusHighlight = ({ status }: NetworkStatusType, theme: Theme) => {
|
const networkStatusHighlight = ({ status }: NetworkStatusType, theme: Theme) => {
|
||||||
switch (status) {
|
switch (status) {
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
||||||
@@ -55,11 +64,6 @@ const networkQualityHighlight = ({ rssi }: NetworkStatusType, theme: Theme) => {
|
|||||||
return theme.palette.success.main;
|
return theme.palette.success.main;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const isWiFi = ({ status }: NetworkStatusType) =>
|
|
||||||
status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED;
|
|
||||||
export const isEthernet = ({ status }: NetworkStatusType) =>
|
|
||||||
status === NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED;
|
|
||||||
|
|
||||||
const dnsServers = ({ dns_ip_1, dns_ip_2 }: NetworkStatusType) => {
|
const dnsServers = ({ dns_ip_1, dns_ip_2 }: NetworkStatusType) => {
|
||||||
if (!dns_ip_1) {
|
if (!dns_ip_1) {
|
||||||
return 'none';
|
return 'none';
|
||||||
@@ -81,6 +85,33 @@ const IPs = (status: NetworkStatusType) => {
|
|||||||
return status.local_ip + ', ' + status.local_ipv6;
|
return status.local_ip + ', ' + status.local_ipv6;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getNetworkStatusText = (
|
||||||
|
status: NetworkConnectionStatus,
|
||||||
|
reconnectCount: number,
|
||||||
|
LL: ReturnType<typeof useI18nContext>['LL']
|
||||||
|
) => {
|
||||||
|
switch (status) {
|
||||||
|
case NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED:
|
||||||
|
return LL.CONNECTED(0) + ' (Ethernet)';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_NO_SHIELD:
|
||||||
|
return LL.INACTIVE(1);
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
||||||
|
return LL.IDLE();
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_NO_SSID_AVAIL:
|
||||||
|
return 'No SSID Available';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECTED:
|
||||||
|
return LL.CONNECTED(0) + ' (WiFi) (' + reconnectCount + ')';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECT_FAILED:
|
||||||
|
return LL.CONNECTED(1) + ' ' + LL.FAILED(0) + ' (' + reconnectCount + ')';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECTION_LOST:
|
||||||
|
return LL.CONNECTED(1) + ' ' + LL.LOST() + ' (' + reconnectCount + ')';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
||||||
|
return LL.DISCONNECTED();
|
||||||
|
default:
|
||||||
|
return LL.UNKNOWN();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const NetworkStatus = () => {
|
const NetworkStatus = () => {
|
||||||
const { data, send: loadData, error } = useRequest(NetworkApi.readNetworkStatus);
|
const { data, send: loadData, error } = useRequest(NetworkApi.readNetworkStatus);
|
||||||
|
|
||||||
@@ -93,51 +124,30 @@ const NetworkStatus = () => {
|
|||||||
|
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
const networkStatus = ({ status }: NetworkStatusType) => {
|
const content = useMemo(() => {
|
||||||
switch (status) {
|
|
||||||
case NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED:
|
|
||||||
return LL.CONNECTED(0) + ' (Ethernet)';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_NO_SHIELD:
|
|
||||||
return LL.INACTIVE(1);
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
|
||||||
return LL.IDLE();
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_NO_SSID_AVAIL:
|
|
||||||
return 'No SSID Available';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECTED:
|
|
||||||
return LL.CONNECTED(0) + ' (WiFi) (' + data.reconnect_count + ')';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECT_FAILED:
|
|
||||||
return (
|
|
||||||
LL.CONNECTED(1) + ' ' + LL.FAILED(0) + ' (' + data.reconnect_count + ')'
|
|
||||||
);
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECTION_LOST:
|
|
||||||
return LL.CONNECTED(1) + ' ' + LL.LOST() + ' (' + data.reconnect_count + ')';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
|
||||||
return LL.DISCONNECTED();
|
|
||||||
default:
|
|
||||||
return LL.UNKNOWN();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const content = () => {
|
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const statusText = getNetworkStatusText(data.status, data.reconnect_count, LL);
|
||||||
|
const statusColor = networkStatusHighlight(data, theme);
|
||||||
|
const qualityColor = networkQualityHighlight(data, theme);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<List>
|
<List>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: networkStatusHighlight(data, theme) }}>
|
<Avatar sx={{ bgcolor: statusColor }}>
|
||||||
{isWiFi(data) && <WifiIcon />}
|
{isWiFi(data) && <WifiIcon />}
|
||||||
{isEthernet(data) && <RouterIcon />}
|
{isEthernet(data) && <RouterIcon />}
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary="Status" secondary={networkStatus(data)} />
|
<ListItemText primary="Status" secondary={statusText} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: networkStatusHighlight(data, theme) }}>
|
<Avatar sx={{ bgcolor: statusColor }}>
|
||||||
<GiteIcon />
|
<GiteIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
@@ -148,13 +158,13 @@ const NetworkStatus = () => {
|
|||||||
<>
|
<>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: networkQualityHighlight(data, theme) }}>
|
<Avatar sx={{ bgcolor: qualityColor }}>
|
||||||
<SettingsInputAntennaIcon />
|
<SettingsInputAntennaIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
primary="SSID (RSSI)"
|
primary="SSID (RSSI)"
|
||||||
secondary={data.ssid + ' (' + data.rssi + ' dBm)'}
|
secondary={`${data.ssid} (${data.rssi} dBm)`}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
@@ -218,9 +228,9 @@ const NetworkStatus = () => {
|
|||||||
)}
|
)}
|
||||||
</List>
|
</List>
|
||||||
);
|
);
|
||||||
};
|
}, [data, error, loadData, LL, theme]);
|
||||||
|
|
||||||
return <SectionContent>{content()}</SectionContent>;
|
return <SectionContent>{content}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default NetworkStatus;
|
export default NetworkStatus;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useContext, useState } from 'react';
|
import { useCallback, useContext, useMemo, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
||||||
@@ -8,10 +8,10 @@ import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
|||||||
import DirectionsBusIcon from '@mui/icons-material/DirectionsBus';
|
import DirectionsBusIcon from '@mui/icons-material/DirectionsBus';
|
||||||
import LogoDevIcon from '@mui/icons-material/LogoDev';
|
import LogoDevIcon from '@mui/icons-material/LogoDev';
|
||||||
import MemoryIcon from '@mui/icons-material/Memory';
|
import MemoryIcon from '@mui/icons-material/Memory';
|
||||||
|
import MonitorHeartIcon from '@mui/icons-material/MonitorHeart';
|
||||||
import PowerSettingsNewIcon from '@mui/icons-material/PowerSettingsNew';
|
import PowerSettingsNewIcon from '@mui/icons-material/PowerSettingsNew';
|
||||||
import RouterIcon from '@mui/icons-material/Router';
|
import RouterIcon from '@mui/icons-material/Router';
|
||||||
import SettingsInputAntennaIcon from '@mui/icons-material/SettingsInputAntenna';
|
import SettingsInputAntennaIcon from '@mui/icons-material/SettingsInputAntenna';
|
||||||
import TimerIcon from '@mui/icons-material/Timer';
|
|
||||||
import WifiIcon from '@mui/icons-material/Wifi';
|
import WifiIcon from '@mui/icons-material/Wifi';
|
||||||
import {
|
import {
|
||||||
Avatar,
|
Avatar,
|
||||||
@@ -37,12 +37,34 @@ import { FormLoader, SectionContent, useLayoutTitle } from 'components';
|
|||||||
import ListMenuItem from 'components/layout/ListMenuItem';
|
import ListMenuItem from 'components/layout/ListMenuItem';
|
||||||
import { AuthenticatedContext } from 'contexts/authentication';
|
import { AuthenticatedContext } from 'contexts/authentication';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
import { NTPSyncStatus, NetworkConnectionStatus } from 'types';
|
import { NTPSyncStatus, NetworkConnectionStatus, SystemStatusCodes } from 'types';
|
||||||
import { useInterval } from 'utils';
|
import { useInterval } from 'utils';
|
||||||
import { formatDateTime } from 'utils/time';
|
import { formatDateTime } from 'utils/time';
|
||||||
|
|
||||||
import SystemMonitor from './SystemMonitor';
|
import SystemMonitor from './SystemMonitor';
|
||||||
|
|
||||||
|
// Pure functions moved outside component to avoid recreation on each render
|
||||||
|
const formatNumber = (num: number) => new Intl.NumberFormat().format(num);
|
||||||
|
|
||||||
|
const formatDurationSec = (
|
||||||
|
duration_sec: number,
|
||||||
|
LL: ReturnType<typeof useI18nContext>['LL']
|
||||||
|
) => {
|
||||||
|
const ms = duration_sec * 1000;
|
||||||
|
const days = Math.trunc(ms / 86400000);
|
||||||
|
const hours = Math.trunc(ms / 3600000) % 24;
|
||||||
|
const minutes = Math.trunc(ms / 60000) % 60;
|
||||||
|
const seconds = Math.trunc(ms / 1000) % 60;
|
||||||
|
|
||||||
|
const parts: string[] = [];
|
||||||
|
if (days) parts.push(LL.NUM_DAYS({ num: days }));
|
||||||
|
if (hours) parts.push(LL.NUM_HOURS({ num: hours }));
|
||||||
|
if (minutes) parts.push(LL.NUM_MINUTES({ num: minutes }));
|
||||||
|
parts.push(LL.NUM_SECONDS({ num: seconds }));
|
||||||
|
|
||||||
|
return parts.join(' ');
|
||||||
|
};
|
||||||
|
|
||||||
const SystemStatus = () => {
|
const SystemStatus = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
@@ -62,7 +84,6 @@ const SystemStatus = () => {
|
|||||||
send: loadData,
|
send: loadData,
|
||||||
error
|
error
|
||||||
} = useRequest(readSystemStatus, {
|
} = useRequest(readSystemStatus, {
|
||||||
initialData: [],
|
|
||||||
async middleware(_, next) {
|
async middleware(_, next) {
|
||||||
if (!restarting) {
|
if (!restarting) {
|
||||||
await next();
|
await next();
|
||||||
@@ -76,51 +97,46 @@ const SystemStatus = () => {
|
|||||||
|
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
const formatDurationSec = (duration_sec: number) => {
|
// Memoize derived status values to avoid recalculation on every render
|
||||||
const days = Math.trunc((duration_sec * 1000) / 86400000);
|
const busStatus = useMemo(() => {
|
||||||
const hours = Math.trunc((duration_sec * 1000) / 3600000) % 24;
|
if (!data) return 'EMS state unknown';
|
||||||
const minutes = Math.trunc((duration_sec * 1000) / 60000) % 60;
|
|
||||||
const seconds = Math.trunc((duration_sec * 1000) / 1000) % 60;
|
|
||||||
|
|
||||||
let formatted = '';
|
|
||||||
if (days) {
|
|
||||||
formatted += LL.NUM_DAYS({ num: days }) + ' ';
|
|
||||||
}
|
|
||||||
if (hours) {
|
|
||||||
formatted += LL.NUM_HOURS({ num: hours }) + ' ';
|
|
||||||
}
|
|
||||||
if (minutes) {
|
|
||||||
formatted += LL.NUM_MINUTES({ num: minutes }) + ' ';
|
|
||||||
}
|
|
||||||
formatted += LL.NUM_SECONDS({ num: seconds });
|
|
||||||
return formatted;
|
|
||||||
};
|
|
||||||
|
|
||||||
function formatNumber(num: number) {
|
|
||||||
return new Intl.NumberFormat().format(num);
|
|
||||||
}
|
|
||||||
|
|
||||||
const busStatus = () => {
|
|
||||||
if (data) {
|
|
||||||
switch (data.bus_status) {
|
switch (data.bus_status) {
|
||||||
case busConnectionStatus.BUS_STATUS_CONNECTED:
|
case busConnectionStatus.BUS_STATUS_CONNECTED:
|
||||||
return (
|
return `EMS ${LL.CONNECTED(0)} (${formatDurationSec(data.bus_uptime, LL)})`;
|
||||||
'EMS ' +
|
|
||||||
LL.CONNECTED(0) +
|
|
||||||
' (' +
|
|
||||||
formatDurationSec(data.bus_uptime) +
|
|
||||||
')'
|
|
||||||
);
|
|
||||||
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
||||||
return 'EMS ' + LL.TX_ISSUES();
|
return 'EMS ' + LL.TX_ISSUES();
|
||||||
case busConnectionStatus.BUS_STATUS_OFFLINE:
|
case busConnectionStatus.BUS_STATUS_OFFLINE:
|
||||||
return 'EMS ' + LL.DISCONNECTED();
|
return 'EMS ' + LL.DISCONNECTED();
|
||||||
}
|
default:
|
||||||
}
|
|
||||||
return 'EMS state unknown';
|
return 'EMS state unknown';
|
||||||
};
|
}
|
||||||
|
}, [data?.bus_status, data?.bus_uptime, LL]);
|
||||||
|
|
||||||
|
// Memoize derived status values to avoid recalculation on every render
|
||||||
|
const systemStatus = useMemo(() => {
|
||||||
|
if (!data) return '??';
|
||||||
|
|
||||||
|
switch (data.status) {
|
||||||
|
case SystemStatusCodes.SYSTEM_STATUS_PENDING_UPLOAD:
|
||||||
|
case SystemStatusCodes.SYSTEM_STATUS_UPLOADING:
|
||||||
|
return LL.WAIT_FIRMWARE();
|
||||||
|
case SystemStatusCodes.SYSTEM_STATUS_ERROR_UPLOAD:
|
||||||
|
return LL.ERROR();
|
||||||
|
case SystemStatusCodes.SYSTEM_STATUS_PENDING_RESTART:
|
||||||
|
case SystemStatusCodes.SYSTEM_STATUS_RESTART_REQUESTED:
|
||||||
|
return LL.RESTARTING_PRE();
|
||||||
|
case SystemStatusCodes.SYSTEM_STATUS_INVALID_GPIO:
|
||||||
|
return LL.GPIO_OF(LL.FAILED(0));
|
||||||
|
default:
|
||||||
|
// SystemStatusCodes.SYSTEM_STATUS_NORMAL
|
||||||
|
return 'OK';
|
||||||
|
}
|
||||||
|
}, [data?.status, LL]);
|
||||||
|
|
||||||
|
const busStatusHighlight = useMemo(() => {
|
||||||
|
if (!data) return theme.palette.warning.main;
|
||||||
|
|
||||||
const busStatusHighlight = () => {
|
|
||||||
switch (data.bus_status) {
|
switch (data.bus_status) {
|
||||||
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
||||||
return theme.palette.warning.main;
|
return theme.palette.warning.main;
|
||||||
@@ -131,27 +147,28 @@ const SystemStatus = () => {
|
|||||||
default:
|
default:
|
||||||
return theme.palette.warning.main;
|
return theme.palette.warning.main;
|
||||||
}
|
}
|
||||||
};
|
}, [data?.bus_status, theme.palette]);
|
||||||
|
|
||||||
|
const ntpStatus = useMemo(() => {
|
||||||
|
if (!data) return LL.UNKNOWN();
|
||||||
|
|
||||||
const ntpStatus = () => {
|
|
||||||
switch (data.ntp_status) {
|
switch (data.ntp_status) {
|
||||||
case NTPSyncStatus.NTP_DISABLED:
|
case NTPSyncStatus.NTP_DISABLED:
|
||||||
return LL.NOT_ENABLED();
|
return LL.NOT_ENABLED();
|
||||||
case NTPSyncStatus.NTP_INACTIVE:
|
case NTPSyncStatus.NTP_INACTIVE:
|
||||||
return LL.INACTIVE(0);
|
return LL.INACTIVE(0);
|
||||||
case NTPSyncStatus.NTP_ACTIVE:
|
case NTPSyncStatus.NTP_ACTIVE:
|
||||||
return (
|
return data.ntp_time
|
||||||
LL.ACTIVE() +
|
? `${LL.ACTIVE()} (${formatDateTime(data.ntp_time)})`
|
||||||
(data.ntp_time !== undefined
|
: LL.ACTIVE();
|
||||||
? ' (' + formatDateTime(data.ntp_time) + ')'
|
|
||||||
: '')
|
|
||||||
);
|
|
||||||
default:
|
default:
|
||||||
return LL.UNKNOWN();
|
return LL.UNKNOWN();
|
||||||
}
|
}
|
||||||
};
|
}, [data?.ntp_status, data?.ntp_time, LL]);
|
||||||
|
|
||||||
|
const ntpStatusHighlight = useMemo(() => {
|
||||||
|
if (!data) return theme.palette.error.main;
|
||||||
|
|
||||||
const ntpStatusHighlight = () => {
|
|
||||||
switch (data.ntp_status) {
|
switch (data.ntp_status) {
|
||||||
case NTPSyncStatus.NTP_DISABLED:
|
case NTPSyncStatus.NTP_DISABLED:
|
||||||
return theme.palette.info.main;
|
return theme.palette.info.main;
|
||||||
@@ -162,9 +179,11 @@ const SystemStatus = () => {
|
|||||||
default:
|
default:
|
||||||
return theme.palette.error.main;
|
return theme.palette.error.main;
|
||||||
}
|
}
|
||||||
};
|
}, [data?.ntp_status, theme.palette]);
|
||||||
|
|
||||||
|
const networkStatusHighlight = useMemo(() => {
|
||||||
|
if (!data) return theme.palette.warning.main;
|
||||||
|
|
||||||
const networkStatusHighlight = () => {
|
|
||||||
switch (data.network_status) {
|
switch (data.network_status) {
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
||||||
@@ -179,9 +198,11 @@ const SystemStatus = () => {
|
|||||||
default:
|
default:
|
||||||
return theme.palette.warning.main;
|
return theme.palette.warning.main;
|
||||||
}
|
}
|
||||||
};
|
}, [data?.network_status, theme.palette]);
|
||||||
|
|
||||||
|
const networkStatus = useMemo(() => {
|
||||||
|
if (!data) return LL.UNKNOWN();
|
||||||
|
|
||||||
const networkStatus = () => {
|
|
||||||
switch (data.network_status) {
|
switch (data.network_status) {
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_NO_SHIELD:
|
case NetworkConnectionStatus.WIFI_STATUS_NO_SHIELD:
|
||||||
return LL.INACTIVE(1);
|
return LL.INACTIVE(1);
|
||||||
@@ -190,24 +211,27 @@ const SystemStatus = () => {
|
|||||||
case NetworkConnectionStatus.WIFI_STATUS_NO_SSID_AVAIL:
|
case NetworkConnectionStatus.WIFI_STATUS_NO_SSID_AVAIL:
|
||||||
return 'No SSID Available';
|
return 'No SSID Available';
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECTED:
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECTED:
|
||||||
return LL.CONNECTED(0) + ' (WiFi, ' + data.wifi_rssi + ' dBm)';
|
return `${LL.CONNECTED(0)} (WiFi, ${data.wifi_rssi} dBm)`;
|
||||||
case NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED:
|
case NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED:
|
||||||
return LL.CONNECTED(0) + ' (Ethernet)';
|
return `${LL.CONNECTED(0)} (Ethernet)`;
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECT_FAILED:
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECT_FAILED:
|
||||||
return LL.CONNECTED(1) + ' ' + LL.FAILED(0);
|
return `${LL.CONNECTED(1)} ${LL.FAILED(0)}`;
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECTION_LOST:
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECTION_LOST:
|
||||||
return LL.CONNECTED(1) + ' ' + LL.LOST();
|
return `${LL.CONNECTED(1)} ${LL.LOST()}`;
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
||||||
return LL.DISCONNECTED();
|
return LL.DISCONNECTED();
|
||||||
default:
|
default:
|
||||||
return LL.UNKNOWN();
|
return LL.UNKNOWN();
|
||||||
}
|
}
|
||||||
};
|
}, [data?.network_status, data?.wifi_rssi, LL]);
|
||||||
|
|
||||||
const activeHighlight = (value: boolean) =>
|
const activeHighlight = useCallback(
|
||||||
value ? theme.palette.success.main : theme.palette.info.main;
|
(value: boolean) =>
|
||||||
|
value ? theme.palette.success.main : theme.palette.info.main,
|
||||||
|
[theme.palette]
|
||||||
|
);
|
||||||
|
|
||||||
const doRestart = async () => {
|
const doRestart = useCallback(async () => {
|
||||||
setConfirmRestart(false);
|
setConfirmRestart(false);
|
||||||
setRestarting(true);
|
setRestarting(true);
|
||||||
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
||||||
@@ -215,13 +239,18 @@ const SystemStatus = () => {
|
|||||||
toast.error(error.message);
|
toast.error(error.message);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
};
|
}, [sendAPI]);
|
||||||
|
|
||||||
const renderRestartDialog = () => (
|
const handleCloseRestartDialog = useCallback(() => {
|
||||||
|
setConfirmRestart(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const renderRestartDialog = useMemo(
|
||||||
|
() => (
|
||||||
<Dialog
|
<Dialog
|
||||||
sx={dialogStyle}
|
sx={dialogStyle}
|
||||||
open={confirmRestart}
|
open={confirmRestart}
|
||||||
onClose={() => setConfirmRestart(false)}
|
onClose={handleCloseRestartDialog}
|
||||||
>
|
>
|
||||||
<DialogTitle>{LL.RESTART()}</DialogTitle>
|
<DialogTitle>{LL.RESTART()}</DialogTitle>
|
||||||
<DialogContent dividers>{LL.RESTART_CONFIRM()}</DialogContent>
|
<DialogContent dividers>{LL.RESTART_CONFIRM()}</DialogContent>
|
||||||
@@ -229,7 +258,7 @@ const SystemStatus = () => {
|
|||||||
<Button
|
<Button
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={() => setConfirmRestart(false)}
|
onClick={handleCloseRestartDialog}
|
||||||
color="secondary"
|
color="secondary"
|
||||||
>
|
>
|
||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
@@ -244,40 +273,80 @@ const SystemStatus = () => {
|
|||||||
</Button>
|
</Button>
|
||||||
</DialogActions>
|
</DialogActions>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
|
),
|
||||||
|
[confirmRestart, handleCloseRestartDialog, doRestart, LL]
|
||||||
);
|
);
|
||||||
|
|
||||||
const content = () => {
|
// Memoize formatted values
|
||||||
|
const firmwareVersion = useMemo(
|
||||||
|
() => `v${data?.emsesp_version || ''}`,
|
||||||
|
[data?.emsesp_version]
|
||||||
|
);
|
||||||
|
|
||||||
|
const uptimeText = useMemo(
|
||||||
|
() => (data ? formatDurationSec(data.uptime, LL) : ''),
|
||||||
|
[data?.uptime, LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const freeMemoryText = useMemo(
|
||||||
|
() => (data ? `${formatNumber(data.free_heap)} KB ${LL.FREE_MEMORY()}` : ''),
|
||||||
|
[data?.free_heap, LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const networkIcon = useMemo(
|
||||||
|
() =>
|
||||||
|
data?.network_status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED
|
||||||
|
? WifiIcon
|
||||||
|
: RouterIcon,
|
||||||
|
[data?.network_status]
|
||||||
|
);
|
||||||
|
|
||||||
|
const mqttStatusText = useMemo(
|
||||||
|
() => (data?.mqtt_status ? LL.CONNECTED(0) : LL.INACTIVE(0)),
|
||||||
|
[data?.mqtt_status, LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const apStatusText = useMemo(
|
||||||
|
() => (data?.ap_status ? LL.ACTIVE() : LL.INACTIVE(0)),
|
||||||
|
[data?.ap_status, LL]
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleRestartClick = useCallback(() => {
|
||||||
|
setConfirmRestart(true);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const content = useMemo(() => {
|
||||||
if (!data || !LL) {
|
if (!data || !LL) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<List sx={{ borderRadius: 3, border: '2px solid grey' }}>
|
<List>
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
icon={BuildIcon}
|
icon={BuildIcon}
|
||||||
bgcolor="#72caf9"
|
bgcolor="#72caf9"
|
||||||
label="EMS-ESP Firmware"
|
label="EMS-ESP Firmware"
|
||||||
text={'v' + data.emsesp_version}
|
text={firmwareVersion}
|
||||||
to="version"
|
to="version"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: '#c5572c', color: 'white' }}>
|
<Avatar sx={{ bgcolor: '#c5572c', color: 'white' }}>
|
||||||
<TimerIcon />
|
<MonitorHeartIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
primary={LL.UPTIME()}
|
primary={LL.STATUS_OF(LL.SYSTEM(0))}
|
||||||
secondary={formatDurationSec(data.uptime)}
|
secondary={`${systemStatus} (${LL.UPTIME()}: ${uptimeText})`}
|
||||||
/>
|
/>
|
||||||
{me.admin && (
|
{me.admin && (
|
||||||
<Button
|
<Button
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="error"
|
color="error"
|
||||||
onClick={() => setConfirmRestart(true)}
|
onClick={handleRestartClick}
|
||||||
>
|
>
|
||||||
{LL.RESTART()}
|
{LL.RESTART()}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -289,29 +358,25 @@ const SystemStatus = () => {
|
|||||||
icon={MemoryIcon}
|
icon={MemoryIcon}
|
||||||
bgcolor="#68374d"
|
bgcolor="#68374d"
|
||||||
label={LL.HARDWARE()}
|
label={LL.HARDWARE()}
|
||||||
text={formatNumber(data.free_heap) + ' KB' + ' ' + LL.FREE_MEMORY()}
|
text={freeMemoryText}
|
||||||
to="/status/hardwarestatus"
|
to="/status/hardwarestatus"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
disabled={!me.admin}
|
disabled={!me.admin}
|
||||||
icon={DirectionsBusIcon}
|
icon={DirectionsBusIcon}
|
||||||
bgcolor={busStatusHighlight()}
|
bgcolor={busStatusHighlight}
|
||||||
label={LL.DATA_TRAFFIC()}
|
label={LL.DATA_TRAFFIC()}
|
||||||
text={busStatus()}
|
text={busStatus}
|
||||||
to="/status/activity"
|
to="/status/activity"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
disabled={!me.admin}
|
disabled={!me.admin}
|
||||||
icon={
|
icon={networkIcon}
|
||||||
data.network_status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED
|
bgcolor={networkStatusHighlight}
|
||||||
? WifiIcon
|
|
||||||
: RouterIcon
|
|
||||||
}
|
|
||||||
bgcolor={networkStatusHighlight()}
|
|
||||||
label={LL.NETWORK(1)}
|
label={LL.NETWORK(1)}
|
||||||
text={networkStatus()}
|
text={networkStatus}
|
||||||
to="/status/network"
|
to="/status/network"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -320,16 +385,16 @@ const SystemStatus = () => {
|
|||||||
icon={DeviceHubIcon}
|
icon={DeviceHubIcon}
|
||||||
bgcolor={activeHighlight(data.mqtt_status)}
|
bgcolor={activeHighlight(data.mqtt_status)}
|
||||||
label="MQTT"
|
label="MQTT"
|
||||||
text={data.mqtt_status ? LL.CONNECTED(0) : LL.INACTIVE(0)}
|
text={mqttStatusText}
|
||||||
to="/status/mqtt"
|
to="/status/mqtt"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
disabled={!me.admin}
|
disabled={!me.admin}
|
||||||
icon={AccessTimeIcon}
|
icon={AccessTimeIcon}
|
||||||
bgcolor={ntpStatusHighlight()}
|
bgcolor={ntpStatusHighlight}
|
||||||
label="NTP"
|
label="NTP"
|
||||||
text={ntpStatus()}
|
text={ntpStatus}
|
||||||
to="/status/ntp"
|
to="/status/ntp"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -338,7 +403,7 @@ const SystemStatus = () => {
|
|||||||
icon={SettingsInputAntennaIcon}
|
icon={SettingsInputAntennaIcon}
|
||||||
bgcolor={activeHighlight(data.ap_status)}
|
bgcolor={activeHighlight(data.ap_status)}
|
||||||
label={LL.ACCESS_POINT(0)}
|
label={LL.ACCESS_POINT(0)}
|
||||||
text={data.ap_status ? LL.ACTIVE() : LL.INACTIVE(0)}
|
text={apStatusText}
|
||||||
to="/status/ap"
|
to="/status/ap"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -352,14 +417,33 @@ const SystemStatus = () => {
|
|||||||
/>
|
/>
|
||||||
</List>
|
</List>
|
||||||
|
|
||||||
{renderRestartDialog()}
|
{renderRestartDialog}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
}, [
|
||||||
|
data,
|
||||||
|
LL,
|
||||||
|
firmwareVersion,
|
||||||
|
uptimeText,
|
||||||
|
freeMemoryText,
|
||||||
|
networkIcon,
|
||||||
|
mqttStatusText,
|
||||||
|
apStatusText,
|
||||||
|
busStatus,
|
||||||
|
busStatusHighlight,
|
||||||
|
networkStatusHighlight,
|
||||||
|
networkStatus,
|
||||||
|
ntpStatusHighlight,
|
||||||
|
ntpStatus,
|
||||||
|
activeHighlight,
|
||||||
|
me.admin,
|
||||||
|
handleRestartClick,
|
||||||
|
error,
|
||||||
|
loadData,
|
||||||
|
renderRestartDialog
|
||||||
|
]);
|
||||||
|
|
||||||
return (
|
return restarting ? <SystemMonitor /> : <SectionContent>{content}</SectionContent>;
|
||||||
<SectionContent>{restarting ? <SystemMonitor /> : content()}</SectionContent>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default SystemStatus;
|
export default SystemStatus;
|
||||||
|
|||||||
@@ -1,4 +1,11 @@
|
|||||||
import { useEffect, useRef, useState } from 'react';
|
import {
|
||||||
|
memo,
|
||||||
|
useCallback,
|
||||||
|
useEffect,
|
||||||
|
useLayoutEffect,
|
||||||
|
useRef,
|
||||||
|
useState
|
||||||
|
} from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import DownloadIcon from '@mui/icons-material/GetApp';
|
import DownloadIcon from '@mui/icons-material/GetApp';
|
||||||
@@ -47,11 +54,6 @@ const LogEntryLine = styled('span')(
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
const topOffset = () =>
|
|
||||||
document.getElementById('log-window')?.getBoundingClientRect().bottom || 0;
|
|
||||||
const leftOffset = () =>
|
|
||||||
document.getElementById('log-window')?.getBoundingClientRect().left || 0;
|
|
||||||
|
|
||||||
const levelLabel = (level: LogLevel) => {
|
const levelLabel = (level: LogLevel) => {
|
||||||
switch (level) {
|
switch (level) {
|
||||||
case LogLevel.ERROR:
|
case LogLevel.ERROR:
|
||||||
@@ -71,6 +73,39 @@ const levelLabel = (level: LogLevel) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const paddedLevelLabel = (level: LogLevel, compact: boolean) => {
|
||||||
|
const label = levelLabel(level);
|
||||||
|
return compact ? ' ' + label[0] : label.padStart(8, '\xa0');
|
||||||
|
};
|
||||||
|
|
||||||
|
const paddedNameLabel = (name: string, compact: boolean) => {
|
||||||
|
const label = '[' + name + ']';
|
||||||
|
return compact ? label : label.padEnd(12, '\xa0');
|
||||||
|
};
|
||||||
|
|
||||||
|
const paddedIDLabel = (id: number, compact: boolean) => {
|
||||||
|
const label = id + ':';
|
||||||
|
return compact ? label : label.padEnd(7, '\xa0');
|
||||||
|
};
|
||||||
|
|
||||||
|
// Memoized log entry component to prevent unnecessary re-renders
|
||||||
|
const LogEntryItem = memo(
|
||||||
|
({ entry, compact }: { entry: LogEntry; compact: boolean }) => {
|
||||||
|
return (
|
||||||
|
<div style={{ font: '13px monospace', whiteSpace: 'nowrap' }}>
|
||||||
|
<span>{entry.t}</span>
|
||||||
|
<span>{paddedLevelLabel(entry.l, compact)} </span>
|
||||||
|
<span>{paddedIDLabel(entry.i, compact)} </span>
|
||||||
|
<span>{paddedNameLabel(entry.n, compact)} </span>
|
||||||
|
<LogEntryLine details={{ level: entry.l }}>{entry.m}</LogEntryLine>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
(prevProps, nextProps) =>
|
||||||
|
prevProps.entry.i === nextProps.entry.i &&
|
||||||
|
prevProps.compact === nextProps.compact
|
||||||
|
);
|
||||||
|
|
||||||
const SystemLog = () => {
|
const SystemLog = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
@@ -102,54 +137,85 @@ const SystemLog = () => {
|
|||||||
const [readOpen, setReadOpen] = useState(false);
|
const [readOpen, setReadOpen] = useState(false);
|
||||||
const [logEntries, setLogEntries] = useState<LogEntry[]>([]);
|
const [logEntries, setLogEntries] = useState<LogEntry[]>([]);
|
||||||
const [autoscroll, setAutoscroll] = useState(true);
|
const [autoscroll, setAutoscroll] = useState(true);
|
||||||
const [lastId, setLastId] = useState<number>(-1);
|
const [boxPosition, setBoxPosition] = useState({ top: 0, left: 0 });
|
||||||
|
|
||||||
const ALPHA_NUMERIC_DASH_REGEX = /^[a-fA-F0-9 ]+$/;
|
const ALPHA_NUMERIC_DASH_REGEX = /^[a-fA-F0-9 ]+$/;
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
const updateFormValue = updateValueDirty(
|
||||||
origData,
|
origData as unknown as Record<string, unknown>,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue as (value: unknown) => void
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Calculate box position after layout
|
||||||
|
useLayoutEffect(() => {
|
||||||
|
const logWindow = document.getElementById('log-window');
|
||||||
|
if (!logWindow) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const updatePosition = () => {
|
||||||
|
const windowElement = document.getElementById('log-window');
|
||||||
|
if (!windowElement) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const rect = windowElement.getBoundingClientRect();
|
||||||
|
setBoxPosition({ top: rect.bottom, left: rect.left });
|
||||||
|
};
|
||||||
|
|
||||||
|
updatePosition();
|
||||||
|
|
||||||
|
// Debounce resize events with requestAnimationFrame
|
||||||
|
let rafId: number;
|
||||||
|
const handleResize = () => {
|
||||||
|
cancelAnimationFrame(rafId);
|
||||||
|
rafId = requestAnimationFrame(updatePosition);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Update position on window resize
|
||||||
|
window.addEventListener('resize', handleResize);
|
||||||
|
const resizeObserver = new ResizeObserver(handleResize);
|
||||||
|
resizeObserver.observe(logWindow);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('resize', handleResize);
|
||||||
|
resizeObserver.disconnect();
|
||||||
|
cancelAnimationFrame(rafId);
|
||||||
|
};
|
||||||
|
}, [data]); // Recalculate when data changes (in case layout shifts)
|
||||||
|
|
||||||
|
// Memoize message handler to avoid recreating on every render
|
||||||
|
const handleLogMessage = useCallback((message: { data: string }) => {
|
||||||
|
const rawData = message.data;
|
||||||
|
const logentry = JSON.parse(rawData) as LogEntry;
|
||||||
|
setLogEntries((log) => {
|
||||||
|
// Skip if this is a duplicate entry (check last entry id)
|
||||||
|
if (log.length > 0) {
|
||||||
|
const lastEntry = log[log.length - 1];
|
||||||
|
if (lastEntry && logentry.i <= lastEntry.i) {
|
||||||
|
return log;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const newLog = [...log, logentry];
|
||||||
|
return newLog;
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
useSSE(fetchLogES, {
|
useSSE(fetchLogES, {
|
||||||
immediate: true,
|
immediate: true,
|
||||||
interceptByGlobalResponded: false
|
interceptByGlobalResponded: false
|
||||||
})
|
})
|
||||||
.onMessage((message: { data: string }) => {
|
.onMessage(handleLogMessage)
|
||||||
const rawData = message.data;
|
|
||||||
const logentry = JSON.parse(rawData) as LogEntry;
|
|
||||||
if (lastId < logentry.i) {
|
|
||||||
setLogEntries((log) => [...log, logentry]);
|
|
||||||
setLastId(logentry.i);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.onError(() => {
|
.onError(() => {
|
||||||
toast.error('No connection to Log service');
|
toast.error('No connection to Log service');
|
||||||
});
|
});
|
||||||
|
|
||||||
const paddedLevelLabel = (level: LogLevel) => {
|
const onDownload = useCallback(() => {
|
||||||
const label = levelLabel(level);
|
const result = logEntries
|
||||||
return data?.compact ? ' ' + label[0] : label.padStart(8, '\xa0');
|
.map((i) => `${i.t} ${levelLabel(i.l)} ${i.i}: [${i.n}] ${i.m}`)
|
||||||
};
|
.join('\n');
|
||||||
|
|
||||||
const paddedNameLabel = (name: string) => {
|
|
||||||
const label = '[' + name + ']';
|
|
||||||
return data?.compact ? label : label.padEnd(12, '\xa0');
|
|
||||||
};
|
|
||||||
|
|
||||||
const paddedIDLabel = (id: number) => {
|
|
||||||
const label = id + ':';
|
|
||||||
return data?.compact ? label : label.padEnd(7, '\xa0');
|
|
||||||
};
|
|
||||||
|
|
||||||
const onDownload = () => {
|
|
||||||
let result = '';
|
|
||||||
for (const i of logEntries) {
|
|
||||||
result +=
|
|
||||||
i.t + ' ' + levelLabel(i.l) + ' ' + i.i + ': [' + i.n + '] ' + i.m + '\n';
|
|
||||||
}
|
|
||||||
const a = document.createElement('a');
|
const a = document.createElement('a');
|
||||||
a.setAttribute(
|
a.setAttribute(
|
||||||
'href',
|
'href',
|
||||||
@@ -159,24 +225,28 @@ const SystemLog = () => {
|
|||||||
document.body.appendChild(a);
|
document.body.appendChild(a);
|
||||||
a.click();
|
a.click();
|
||||||
document.body.removeChild(a);
|
document.body.removeChild(a);
|
||||||
};
|
}, [logEntries]);
|
||||||
|
|
||||||
const saveSettings = async () => {
|
const saveSettings = useCallback(async () => {
|
||||||
await saveData();
|
await saveData();
|
||||||
};
|
}, [saveData]);
|
||||||
|
|
||||||
// handle scrolling
|
// handle scrolling - optimized to only scroll when needed
|
||||||
const ref = useRef<HTMLDivElement>(null);
|
const ref = useRef<HTMLDivElement>(null);
|
||||||
|
const logWindowRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (logEntries.length && autoscroll) {
|
if (logEntries.length && autoscroll) {
|
||||||
ref.current?.scrollIntoView({
|
const container = logWindowRef.current;
|
||||||
behavior: 'smooth',
|
if (container) {
|
||||||
block: 'end'
|
requestAnimationFrame(() => {
|
||||||
|
container.scrollTop = container.scrollHeight;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, [logEntries.length]);
|
}
|
||||||
|
}, [logEntries.length, autoscroll]);
|
||||||
|
|
||||||
const sendReadCommand = () => {
|
const sendReadCommand = useCallback(() => {
|
||||||
if (readValue === '') {
|
if (readValue === '') {
|
||||||
setReadOpen(!readOpen);
|
setReadOpen(!readOpen);
|
||||||
return;
|
return;
|
||||||
@@ -187,7 +257,7 @@ const SystemLog = () => {
|
|||||||
setReadOpen(false);
|
setReadOpen(false);
|
||||||
setReadValue('');
|
setReadValue('');
|
||||||
}
|
}
|
||||||
};
|
}, [readValue, readOpen, send]);
|
||||||
|
|
||||||
const content = () => {
|
const content = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
@@ -232,6 +302,8 @@ const SystemLog = () => {
|
|||||||
<MenuItem value={50}>50</MenuItem>
|
<MenuItem value={50}>50</MenuItem>
|
||||||
<MenuItem value={75}>75</MenuItem>
|
<MenuItem value={75}>75</MenuItem>
|
||||||
<MenuItem value={100}>100</MenuItem>
|
<MenuItem value={100}>100</MenuItem>
|
||||||
|
<MenuItem value={500}>500</MenuItem>
|
||||||
|
<MenuItem value={1000}>1000</MenuItem>
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
@@ -279,6 +351,7 @@ const SystemLog = () => {
|
|||||||
>
|
>
|
||||||
<IconButton
|
<IconButton
|
||||||
disableRipple
|
disableRipple
|
||||||
|
aria-label={LL.CANCEL()}
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setReadOpen(false);
|
setReadOpen(false);
|
||||||
setReadValue('');
|
setReadValue('');
|
||||||
@@ -304,7 +377,7 @@ const SystemLog = () => {
|
|||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
{data.developer_mode && (
|
{data.developer_mode && (
|
||||||
<IconButton onClick={sendReadCommand}>
|
<IconButton onClick={sendReadCommand} aria-label={LL.EXECUTE()}>
|
||||||
<PlayArrowIcon color="primary" />
|
<PlayArrowIcon color="primary" />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
)}
|
)}
|
||||||
@@ -326,27 +399,20 @@ const SystemLog = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
<Box
|
<Box
|
||||||
|
ref={logWindowRef}
|
||||||
sx={{
|
sx={{
|
||||||
backgroundColor: 'black',
|
backgroundColor: 'black',
|
||||||
overflowY: 'scroll',
|
overflowY: 'scroll',
|
||||||
position: 'absolute',
|
position: 'absolute',
|
||||||
right: 18,
|
right: 18,
|
||||||
bottom: 18,
|
bottom: 18,
|
||||||
left: () => leftOffset(),
|
left: boxPosition.left,
|
||||||
top: () => topOffset(),
|
top: boxPosition.top,
|
||||||
p: 1
|
p: 1
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{logEntries.map((e) => (
|
{logEntries.map((e) => (
|
||||||
<div key={e.i} style={{ font: '14px monospace', whiteSpace: 'nowrap' }}>
|
<LogEntryItem key={e.i} entry={e} compact={data.compact} />
|
||||||
<span>{e.t}</span>
|
|
||||||
<span>{paddedLevelLabel(e.l)} </span>
|
|
||||||
<span>{paddedIDLabel(e.i)} </span>
|
|
||||||
<span>{paddedNameLabel(e.n)} </span>
|
|
||||||
<LogEntryLine details={{ level: e.l }} key={e.i}>
|
|
||||||
{e.m}
|
|
||||||
</LogEntryLine>
|
|
||||||
</div>
|
|
||||||
))}
|
))}
|
||||||
|
|
||||||
<div ref={ref} />
|
<div ref={ref} />
|
||||||
|
|||||||
@@ -1,12 +1,11 @@
|
|||||||
import { useState } from 'react';
|
import { useCallback, useMemo, useRef, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import { Box, Button, Dialog, DialogContent, Typography } from '@mui/material';
|
import { Box, Button, Typography } from '@mui/material';
|
||||||
|
|
||||||
import { callAction } from 'api/app';
|
import { callAction } from 'api/app';
|
||||||
import { readSystemStatus } from 'api/system';
|
import { readSystemStatus } from 'api/system';
|
||||||
|
|
||||||
import { dialogStyle } from 'CustomTheme';
|
|
||||||
import { useRequest } from 'alova/client';
|
import { useRequest } from 'alova/client';
|
||||||
import MessageBox from 'components/MessageBox';
|
import MessageBox from 'components/MessageBox';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
@@ -17,11 +16,9 @@ import { LinearProgressWithLabel } from '../../components/upload/LinearProgressW
|
|||||||
|
|
||||||
const SystemMonitor = () => {
|
const SystemMonitor = () => {
|
||||||
const [errorMessage, setErrorMessage] = useState<string>();
|
const [errorMessage, setErrorMessage] = useState<string>();
|
||||||
|
const hasInitialized = useRef(false);
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
let count = 0;
|
|
||||||
|
|
||||||
const { send: setSystemStatus } = useRequest(
|
const { send: setSystemStatus } = useRequest(
|
||||||
(status: string) => callAction({ action: 'systemStatus', param: status }),
|
(status: string) => callAction({ action: 'systemStatus', param: status }),
|
||||||
{
|
{
|
||||||
@@ -32,10 +29,12 @@ const SystemMonitor = () => {
|
|||||||
const { data, send } = useRequest(readSystemStatus, {
|
const { data, send } = useRequest(readSystemStatus, {
|
||||||
force: true,
|
force: true,
|
||||||
async middleware(_, next) {
|
async middleware(_, next) {
|
||||||
if (count++ >= 1) {
|
// Skip first request to allow AsyncWS to send its response
|
||||||
// skip first request (1 second) to allow AsyncWS to send its response
|
if (!hasInitialized.current) {
|
||||||
await next();
|
hasInitialized.current = true;
|
||||||
|
return; // Don't await next() on first call
|
||||||
}
|
}
|
||||||
|
await next();
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.onSuccess((event) => {
|
.onSuccess((event) => {
|
||||||
@@ -58,40 +57,87 @@ const SystemMonitor = () => {
|
|||||||
void send();
|
void send();
|
||||||
}, 1000); // check every 1 second
|
}, 1000); // check every 1 second
|
||||||
|
|
||||||
const onCancel = async () => {
|
const { statusMessage, isUploading, progressValue } = useMemo(() => {
|
||||||
setErrorMessage(undefined);
|
const status = data?.status;
|
||||||
await setSystemStatus(
|
|
||||||
SystemStatusCodes.SYSTEM_STATUS_NORMAL as unknown as string
|
const message =
|
||||||
);
|
status && status >= SystemStatusCodes.SYSTEM_STATUS_UPLOADING
|
||||||
document.location.href = '/';
|
? LL.WAIT_FIRMWARE()
|
||||||
|
: status === SystemStatusCodes.SYSTEM_STATUS_PENDING_RESTART
|
||||||
|
? LL.APPLICATION_RESTARTING()
|
||||||
|
: status === SystemStatusCodes.SYSTEM_STATUS_NORMAL
|
||||||
|
? LL.RESTARTING_PRE()
|
||||||
|
: status === SystemStatusCodes.SYSTEM_STATUS_ERROR_UPLOAD
|
||||||
|
? 'Upload Failed'
|
||||||
|
: LL.RESTARTING_POST();
|
||||||
|
|
||||||
|
const uploading =
|
||||||
|
status !== undefined && status >= SystemStatusCodes.SYSTEM_STATUS_UPLOADING;
|
||||||
|
const progress =
|
||||||
|
uploading && status
|
||||||
|
? Math.round(status - SystemStatusCodes.SYSTEM_STATUS_UPLOADING)
|
||||||
|
: 0;
|
||||||
|
|
||||||
|
return {
|
||||||
|
statusMessage: message,
|
||||||
|
isUploading: uploading,
|
||||||
|
progressValue: progress
|
||||||
};
|
};
|
||||||
|
}, [data?.status, LL]);
|
||||||
|
|
||||||
|
const onCancel = useCallback(async () => {
|
||||||
|
setErrorMessage(undefined);
|
||||||
|
await setSystemStatus(String(SystemStatusCodes.SYSTEM_STATUS_NORMAL));
|
||||||
|
document.location.href = '/';
|
||||||
|
}, [setSystemStatus]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog fullWidth={true} sx={dialogStyle} open={true}>
|
<Box
|
||||||
<DialogContent dividers>
|
sx={{
|
||||||
<Box m={0} py={0} display="flex" alignItems="center" flexDirection="column">
|
position: 'fixed',
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
width: '100vw',
|
||||||
|
height: '100vh',
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'center'
|
||||||
|
// backdropFilter: 'blur(8px)'
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Box
|
||||||
|
sx={{
|
||||||
|
width: '30%',
|
||||||
|
minWidth: '300px',
|
||||||
|
maxWidth: '500px',
|
||||||
|
backgroundColor: '#393939',
|
||||||
|
border: 2,
|
||||||
|
borderColor: '#565656',
|
||||||
|
borderRadius: '8px',
|
||||||
|
boxShadow: '0 8px 32px rgba(0, 0, 0, 0.3)',
|
||||||
|
p: 3
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Box display="flex" alignItems="center" flexDirection="column">
|
||||||
|
<img
|
||||||
|
src="/app/icon.png"
|
||||||
|
alt="EMS-ESP"
|
||||||
|
style={{ width: '40px', height: '40px', marginBottom: '16px' }}
|
||||||
|
/>
|
||||||
<Typography
|
<Typography
|
||||||
color="secondary"
|
color="secondary"
|
||||||
variant="h6"
|
variant="h6"
|
||||||
fontWeight={400}
|
fontWeight={400}
|
||||||
textAlign="center"
|
textAlign="center"
|
||||||
>
|
>
|
||||||
{data?.status >= SystemStatusCodes.SYSTEM_STATUS_UPLOADING
|
{statusMessage}
|
||||||
? LL.WAIT_FIRMWARE()
|
|
||||||
: data?.status === SystemStatusCodes.SYSTEM_STATUS_PENDING_RESTART
|
|
||||||
? LL.APPLICATION_RESTARTING()
|
|
||||||
: data?.status === SystemStatusCodes.SYSTEM_STATUS_NORMAL
|
|
||||||
? LL.RESTARTING_PRE()
|
|
||||||
: data?.status === SystemStatusCodes.SYSTEM_STATUS_ERROR_UPLOAD
|
|
||||||
? 'Upload Failed'
|
|
||||||
: LL.RESTARTING_POST()}
|
|
||||||
</Typography>
|
</Typography>
|
||||||
|
|
||||||
{errorMessage ? (
|
{errorMessage ? (
|
||||||
<MessageBox my={2} level="error" message={errorMessage}>
|
<MessageBox level="error" message={errorMessage}>
|
||||||
<Button
|
<Button
|
||||||
size="small"
|
|
||||||
sx={{ ml: 2 }}
|
sx={{ ml: 2 }}
|
||||||
|
size="small"
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="contained"
|
variant="contained"
|
||||||
color="error"
|
color="error"
|
||||||
@@ -105,20 +151,16 @@ const SystemMonitor = () => {
|
|||||||
<Typography mt={2} variant="h6" fontWeight={400} textAlign="center">
|
<Typography mt={2} variant="h6" fontWeight={400} textAlign="center">
|
||||||
{LL.PLEASE_WAIT()}…
|
{LL.PLEASE_WAIT()}…
|
||||||
</Typography>
|
</Typography>
|
||||||
{data && data.status >= SystemStatusCodes.SYSTEM_STATUS_UPLOADING && (
|
{isUploading && (
|
||||||
<Box width="100%" pl={2} pr={2} py={2}>
|
<Box width="100%" pl={2} pr={2} py={2}>
|
||||||
<LinearProgressWithLabel
|
<LinearProgressWithLabel value={progressValue} />
|
||||||
value={Math.round(
|
|
||||||
data?.status - SystemStatusCodes.SYSTEM_STATUS_UPLOADING
|
|
||||||
)}
|
|
||||||
/>
|
|
||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
</DialogContent>
|
</Box>
|
||||||
</Dialog>
|
</Box>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,9 +1,9 @@
|
|||||||
import { memo } from 'react';
|
import { type FC, type PropsWithChildren, memo } from 'react';
|
||||||
|
|
||||||
import { Box } from '@mui/material';
|
import { Box } from '@mui/material';
|
||||||
import type { BoxProps } from '@mui/material';
|
import type { BoxProps } from '@mui/material';
|
||||||
|
|
||||||
const ButtonRow = memo<BoxProps>(({ children, ...rest }) => (
|
const ButtonRow: FC<PropsWithChildren<BoxProps>> = memo(({ children, ...rest }) => (
|
||||||
<Box
|
<Box
|
||||||
sx={{
|
sx={{
|
||||||
'& button, & a, & .MuiCard-root': {
|
'& button, & a, & .MuiCard-root': {
|
||||||
@@ -19,6 +19,4 @@ const ButtonRow = memo<BoxProps>(({ children, ...rest }) => (
|
|||||||
</Box>
|
</Box>
|
||||||
));
|
));
|
||||||
|
|
||||||
ButtonRow.displayName = 'ButtonRow';
|
|
||||||
|
|
||||||
export default ButtonRow;
|
export default ButtonRow;
|
||||||
|
|||||||
@@ -1,22 +1,7 @@
|
|||||||
import { Tooltip, type TooltipProps, styled, tooltipClasses } from '@mui/material';
|
import { Tooltip, type TooltipProps } from '@mui/material';
|
||||||
|
|
||||||
export const ButtonTooltip = styled(({ className, ...props }: TooltipProps) => (
|
export const ButtonTooltip = ({ children, ...props }: TooltipProps) => (
|
||||||
<Tooltip
|
<Tooltip {...props}>{children}</Tooltip>
|
||||||
{...props}
|
);
|
||||||
placement="top"
|
|
||||||
arrow
|
|
||||||
classes={{ ...(className && { popper: className }) }}
|
|
||||||
/>
|
|
||||||
))(({ theme }) => ({
|
|
||||||
[`& .${tooltipClasses.arrow}`]: {
|
|
||||||
color: theme.palette.success.main
|
|
||||||
},
|
|
||||||
[`& .${tooltipClasses.tooltip}`]: {
|
|
||||||
backgroundColor: theme.palette.success.main,
|
|
||||||
color: 'rgba(0, 0, 0, 0.87)',
|
|
||||||
boxShadow: theme.shadows[1],
|
|
||||||
fontSize: 10
|
|
||||||
}
|
|
||||||
}));
|
|
||||||
|
|
||||||
export default ButtonTooltip;
|
export default ButtonTooltip;
|
||||||
|
|||||||
@@ -1,38 +1,35 @@
|
|||||||
import type { FC } from 'react';
|
import { type FC, type PropsWithChildren, memo, useMemo } from 'react';
|
||||||
|
|
||||||
import CheckCircleOutlineOutlinedIcon from '@mui/icons-material/CheckCircleOutlineOutlined';
|
import CheckCircleOutlineOutlinedIcon from '@mui/icons-material/CheckCircleOutlineOutlined';
|
||||||
import ErrorIcon from '@mui/icons-material/Error';
|
import ErrorIcon from '@mui/icons-material/Error';
|
||||||
import InfoOutlinedIcon from '@mui/icons-material/InfoOutlined';
|
import InfoOutlinedIcon from '@mui/icons-material/InfoOutlined';
|
||||||
import ReportProblemOutlinedIcon from '@mui/icons-material/ReportProblemOutlined';
|
import ReportProblemOutlinedIcon from '@mui/icons-material/ReportProblemOutlined';
|
||||||
import { Box, Typography, useTheme } from '@mui/material';
|
import { Box, Typography, useTheme } from '@mui/material';
|
||||||
import type { BoxProps, SvgIconProps, Theme } from '@mui/material';
|
import type { BoxProps, SvgIconProps } from '@mui/material';
|
||||||
|
|
||||||
type MessageBoxLevel = 'warning' | 'success' | 'info' | 'error';
|
type MessageBoxLevel = 'warning' | 'success' | 'info' | 'error';
|
||||||
|
|
||||||
export interface MessageBoxProps extends BoxProps {
|
export interface MessageBoxProps extends BoxProps {
|
||||||
level: MessageBoxLevel;
|
level: MessageBoxLevel;
|
||||||
message?: string;
|
message?: string;
|
||||||
|
children?: React.ReactNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
const LEVEL_ICONS: {
|
const LEVEL_ICONS: Record<MessageBoxLevel, React.ComponentType<SvgIconProps>> = {
|
||||||
[type in MessageBoxLevel]: React.ComponentType<SvgIconProps>;
|
|
||||||
} = {
|
|
||||||
success: CheckCircleOutlineOutlinedIcon,
|
success: CheckCircleOutlineOutlinedIcon,
|
||||||
info: InfoOutlinedIcon,
|
info: InfoOutlinedIcon,
|
||||||
warning: ReportProblemOutlinedIcon,
|
warning: ReportProblemOutlinedIcon,
|
||||||
error: ErrorIcon
|
error: ErrorIcon
|
||||||
};
|
};
|
||||||
|
|
||||||
const LEVEL_BACKGROUNDS: {
|
const LEVEL_PALETTE_PATHS: Record<MessageBoxLevel, string> = {
|
||||||
[type in MessageBoxLevel]: (theme: Theme) => string;
|
success: 'success.dark',
|
||||||
} = {
|
info: 'info.main',
|
||||||
success: (theme: Theme) => theme.palette.success.dark,
|
warning: 'warning.dark',
|
||||||
info: (theme: Theme) => theme.palette.info.main,
|
error: 'error.dark'
|
||||||
warning: (theme: Theme) => theme.palette.warning.dark,
|
|
||||||
error: (theme: Theme) => theme.palette.error.dark
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const MessageBox: FC<MessageBoxProps> = ({
|
const MessageBox: FC<PropsWithChildren<MessageBoxProps>> = ({
|
||||||
level,
|
level,
|
||||||
message,
|
message,
|
||||||
sx,
|
sx,
|
||||||
@@ -40,25 +37,38 @@ const MessageBox: FC<MessageBoxProps> = ({
|
|||||||
...rest
|
...rest
|
||||||
}) => {
|
}) => {
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
|
const { Icon, backgroundColor } = useMemo(() => {
|
||||||
const Icon = LEVEL_ICONS[level];
|
const Icon = LEVEL_ICONS[level];
|
||||||
const backgroundColor = LEVEL_BACKGROUNDS[level](theme);
|
const palettePath = LEVEL_PALETTE_PATHS[level];
|
||||||
const color = 'white';
|
const [key, shade] = palettePath.split('.') as [
|
||||||
|
keyof typeof theme.palette,
|
||||||
|
string
|
||||||
|
];
|
||||||
|
const paletteKey = theme.palette[key] as unknown as Record<string, string>;
|
||||||
|
const backgroundColor = paletteKey[shade];
|
||||||
|
|
||||||
|
return { Icon, backgroundColor };
|
||||||
|
}, [level, theme]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box
|
<Box
|
||||||
p={2}
|
p={2}
|
||||||
display="flex"
|
display="flex"
|
||||||
alignItems="center"
|
alignItems="center"
|
||||||
borderRadius={1}
|
borderRadius={1}
|
||||||
sx={{ backgroundColor, color, ...sx }}
|
sx={{ backgroundColor, color: 'white', ...sx }}
|
||||||
{...rest}
|
{...rest}
|
||||||
>
|
>
|
||||||
<Icon />
|
<Icon />
|
||||||
|
{(message || children) && (
|
||||||
<Typography sx={{ ml: 2 }} variant="body1">
|
<Typography sx={{ ml: 2 }} variant="body1">
|
||||||
{message ?? ''}
|
{message}
|
||||||
</Typography>
|
|
||||||
{children}
|
{children}
|
||||||
|
</Typography>
|
||||||
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default MessageBox;
|
export default memo(MessageBox);
|
||||||
|
|||||||
@@ -1,33 +1,28 @@
|
|||||||
|
import { memo } from 'react';
|
||||||
import type { FC } from 'react';
|
import type { FC } from 'react';
|
||||||
|
|
||||||
import { Divider, Paper } from '@mui/material';
|
import { Paper } from '@mui/material';
|
||||||
|
import type { SxProps, Theme } from '@mui/material/styles';
|
||||||
|
|
||||||
import type { RequiredChildrenProps } from 'utils';
|
import type { RequiredChildrenProps } from 'utils';
|
||||||
|
|
||||||
interface SectionContentProps extends RequiredChildrenProps {
|
interface SectionContentProps extends RequiredChildrenProps {
|
||||||
title?: string;
|
|
||||||
id?: string;
|
id?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const SectionContent: FC<SectionContentProps> = (props) => {
|
// Extract styles to avoid recreation on every render
|
||||||
const { children, title, id } = props;
|
const paperStyles: SxProps<Theme> = {
|
||||||
return (
|
p: 1.5,
|
||||||
<Paper id={id} sx={{ p: 2, m: 2 }}>
|
m: 1.5,
|
||||||
{title && (
|
borderRadius: 3,
|
||||||
<Divider
|
border: '1px solid rgb(65, 65, 65)'
|
||||||
sx={{
|
};
|
||||||
pb: 2,
|
|
||||||
borderColor: 'primary.main',
|
const SectionContent: FC<SectionContentProps> = ({ children, id }) => (
|
||||||
fontSize: 20,
|
<Paper id={id} sx={paperStyles}>
|
||||||
color: 'primary.main'
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{title}
|
|
||||||
</Divider>
|
|
||||||
)}
|
|
||||||
{children}
|
{children}
|
||||||
</Paper>
|
</Paper>
|
||||||
);
|
);
|
||||||
};
|
|
||||||
|
|
||||||
export default SectionContent;
|
// Memoize to prevent unnecessary re-renders
|
||||||
|
export default memo(SectionContent);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
// Optimized exports - use direct exports to reduce bundle size
|
// use direct exports to reduce bundle size
|
||||||
export { default as SectionContent } from './SectionContent';
|
export { default as SectionContent } from './SectionContent';
|
||||||
export { default as ButtonRow } from './ButtonRow';
|
export { default as ButtonRow } from './ButtonRow';
|
||||||
export { default as MessageBox } from './MessageBox';
|
export { default as MessageBox } from './MessageBox';
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { memo } from 'react';
|
||||||
import type { FC } from 'react';
|
import type { FC } from 'react';
|
||||||
|
|
||||||
import { FormControlLabel } from '@mui/material';
|
import { FormControlLabel } from '@mui/material';
|
||||||
@@ -9,4 +10,4 @@ const BlockFormControlLabel: FC<FormControlLabelProps> = (props) => (
|
|||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
export default BlockFormControlLabel;
|
export default memo(BlockFormControlLabel);
|
||||||
|
|||||||
@@ -1,4 +1,6 @@
|
|||||||
import { type ChangeEventHandler, useContext } from 'react';
|
import { memo, useCallback, useContext, useMemo } from 'react';
|
||||||
|
import type { ChangeEventHandler } from 'react';
|
||||||
|
import type { CSSProperties } from 'react';
|
||||||
|
|
||||||
import { MenuItem, TextField } from '@mui/material';
|
import { MenuItem, TextField } from '@mui/material';
|
||||||
|
|
||||||
@@ -17,73 +19,66 @@ import { I18nContext } from 'i18n/i18n-react';
|
|||||||
import type { Locales } from 'i18n/i18n-types';
|
import type { Locales } from 'i18n/i18n-types';
|
||||||
import { loadLocaleAsync } from 'i18n/i18n-util.async';
|
import { loadLocaleAsync } from 'i18n/i18n-util.async';
|
||||||
|
|
||||||
const LanguageSelector = () => {
|
const flagStyle: CSSProperties = { width: 16, verticalAlign: 'middle' };
|
||||||
const { setLocale, locale } = useContext(I18nContext);
|
|
||||||
|
|
||||||
const onLocaleSelected: ChangeEventHandler<HTMLInputElement> = async ({
|
interface LanguageOption {
|
||||||
target
|
key: Locales;
|
||||||
}) => {
|
flag: string;
|
||||||
|
label: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const LANGUAGE_OPTIONS: LanguageOption[] = [
|
||||||
|
{ key: 'cz', flag: CZflag, label: 'CZ' },
|
||||||
|
{ key: 'de', flag: DEflag, label: 'DE' },
|
||||||
|
{ key: 'en', flag: GBflag, label: 'EN' },
|
||||||
|
{ key: 'fr', flag: FRflag, label: 'FR' },
|
||||||
|
{ key: 'it', flag: ITflag, label: 'IT' },
|
||||||
|
{ key: 'nl', flag: NLflag, label: 'NL' },
|
||||||
|
{ key: 'no', flag: NOflag, label: 'NO' },
|
||||||
|
{ key: 'pl', flag: PLflag, label: 'PL' },
|
||||||
|
{ key: 'sk', flag: SKflag, label: 'SK' },
|
||||||
|
{ key: 'sv', flag: SVflag, label: 'SV' },
|
||||||
|
{ key: 'tr', flag: TRflag, label: 'TR' }
|
||||||
|
];
|
||||||
|
|
||||||
|
const LanguageSelector = () => {
|
||||||
|
const { setLocale, locale, LL } = useContext(I18nContext);
|
||||||
|
|
||||||
|
const onLocaleSelected: ChangeEventHandler<HTMLInputElement> = useCallback(
|
||||||
|
async ({ target }) => {
|
||||||
const loc = target.value as Locales;
|
const loc = target.value as Locales;
|
||||||
localStorage.setItem('lang', loc);
|
localStorage.setItem('lang', loc);
|
||||||
await loadLocaleAsync(loc);
|
await loadLocaleAsync(loc);
|
||||||
setLocale(loc);
|
setLocale(loc);
|
||||||
};
|
},
|
||||||
|
[setLocale]
|
||||||
|
);
|
||||||
|
|
||||||
|
// Memoize menu items to prevent recreation on every render
|
||||||
|
const menuItems = useMemo(
|
||||||
|
() =>
|
||||||
|
LANGUAGE_OPTIONS.map(({ key, flag, label }) => (
|
||||||
|
<MenuItem key={key} value={key}>
|
||||||
|
<img src={flag} style={flagStyle} alt={label} />
|
||||||
|
{label}
|
||||||
|
</MenuItem>
|
||||||
|
)),
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<TextField
|
<TextField
|
||||||
name="locale"
|
name="locale"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
|
aria-label={LL.LANGUAGE()}
|
||||||
value={locale}
|
value={locale}
|
||||||
onChange={onLocaleSelected}
|
onChange={onLocaleSelected}
|
||||||
size="small"
|
size="small"
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
<MenuItem key="cz" value="cz">
|
{menuItems}
|
||||||
<img src={CZflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
CZ
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="de" value="de">
|
|
||||||
<img src={DEflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
DE
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="en" value="en">
|
|
||||||
<img src={GBflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
EN
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="fr" value="fr">
|
|
||||||
<img src={FRflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
FR
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="it" value="it">
|
|
||||||
<img src={ITflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
IT
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="nl" value="nl">
|
|
||||||
<img src={NLflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
NL
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="no" value="no">
|
|
||||||
<img src={NOflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
NO
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="pl" value="pl">
|
|
||||||
<img src={PLflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
PL
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="sk" value="sk">
|
|
||||||
<img src={SKflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
SK
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="sv" value="sv">
|
|
||||||
<img src={SVflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
SV
|
|
||||||
</MenuItem>
|
|
||||||
<MenuItem key="tr" value="tr">
|
|
||||||
<img src={TRflag} style={{ width: 16, verticalAlign: 'middle' }} />
|
|
||||||
TR
|
|
||||||
</MenuItem>
|
|
||||||
</TextField>
|
</TextField>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default LanguageSelector;
|
export default memo(LanguageSelector);
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useState } from 'react';
|
import { memo, useCallback, useState } from 'react';
|
||||||
import type { FC } from 'react';
|
import type { FC } from 'react';
|
||||||
|
|
||||||
import VisibilityIcon from '@mui/icons-material/Visibility';
|
import VisibilityIcon from '@mui/icons-material/Visibility';
|
||||||
@@ -13,6 +13,10 @@ type ValidatedPasswordFieldProps = Omit<ValidatedTextFieldProps, 'type'>;
|
|||||||
const ValidatedPasswordField: FC<ValidatedPasswordFieldProps> = ({ ...props }) => {
|
const ValidatedPasswordField: FC<ValidatedPasswordFieldProps> = ({ ...props }) => {
|
||||||
const [showPassword, setShowPassword] = useState<boolean>(false);
|
const [showPassword, setShowPassword] = useState<boolean>(false);
|
||||||
|
|
||||||
|
const togglePasswordVisibility = useCallback(() => {
|
||||||
|
setShowPassword((prev) => !prev);
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
{...props}
|
{...props}
|
||||||
@@ -21,7 +25,11 @@ const ValidatedPasswordField: FC<ValidatedPasswordFieldProps> = ({ ...props }) =
|
|||||||
input: {
|
input: {
|
||||||
endAdornment: (
|
endAdornment: (
|
||||||
<InputAdornment position="end">
|
<InputAdornment position="end">
|
||||||
<IconButton onClick={() => setShowPassword(!showPassword)} edge="end">
|
<IconButton
|
||||||
|
onClick={togglePasswordVisibility}
|
||||||
|
edge="end"
|
||||||
|
aria-label="Password visibility"
|
||||||
|
>
|
||||||
{showPassword ? <VisibilityIcon /> : <VisibilityOffIcon />}
|
{showPassword ? <VisibilityIcon /> : <VisibilityOffIcon />}
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</InputAdornment>
|
</InputAdornment>
|
||||||
@@ -32,4 +40,4 @@ const ValidatedPasswordField: FC<ValidatedPasswordFieldProps> = ({ ...props }) =
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default ValidatedPasswordField;
|
export default memo(ValidatedPasswordField);
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user