mirror of
https://github.com/wanderer-industries/wanderer
synced 2025-11-03 07:57:05 +00:00
Compare commits
785 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
959041be52 | ||
|
|
3319520179 | ||
|
|
580fcf3657 | ||
|
|
53dae7c520 | ||
|
|
6d59d709f1 | ||
|
|
4343e9070c | ||
|
|
b62373fb5f | ||
|
|
3da98f8e56 | ||
|
|
494d24952e | ||
|
|
8a6b17bd7b | ||
|
|
d2e859a74e | ||
|
|
4a78d55d22 | ||
|
|
dc252b8c4b | ||
|
|
c433205e89 | ||
|
|
d6bc5b57b1 | ||
|
|
280a286266 | ||
|
|
d5c18b5de3 | ||
|
|
7452e5d011 | ||
|
|
71674b0d52 | ||
|
|
5b4824bd5d | ||
|
|
deda16a7da | ||
|
|
0b7c067de7 | ||
|
|
0d0db8c129 | ||
|
|
9f1b7994a3 | ||
|
|
378df0ac70 | ||
|
|
0e4a132f69 | ||
|
|
631746375d | ||
|
|
7dc01dad54 | ||
|
|
8a9807d3e5 | ||
|
|
39df3c97ce | ||
|
|
46c1ccdfcc | ||
|
|
8817536038 | ||
|
|
c3bb23a6ee | ||
|
|
7e9c4c575e | ||
|
|
5a70eee91e | ||
|
|
228f6990a1 | ||
|
|
d80ed0e70e | ||
|
|
4576c75737 | ||
|
|
67764faaa7 | ||
|
|
91dd0b27ae | ||
|
|
99dcf49fbc | ||
|
|
6fb3edbfd6 | ||
|
|
26f13ce857 | ||
|
|
e9b475c0a8 | ||
|
|
7752010092 | ||
|
|
d3705b3ed7 | ||
|
|
1394e2897e | ||
|
|
5117a1c5af | ||
|
|
3c62403f33 | ||
|
|
a4760f5162 | ||
|
|
b071070431 | ||
|
|
3bcb9628e7 | ||
|
|
e62c4cf5bf | ||
|
|
af46962ce4 | ||
|
|
0b0967830b | ||
|
|
172251a208 | ||
|
|
8a6fb63d55 | ||
|
|
9652959e5e | ||
|
|
825ef46d41 | ||
|
|
ad9f7c6b95 | ||
|
|
b960b5c149 | ||
|
|
0f092d21f9 | ||
|
|
031576caa6 | ||
|
|
7a97a96c42 | ||
|
|
2efb2daba0 | ||
|
|
4374c39924 | ||
|
|
15711495c7 | ||
|
|
236f803427 | ||
|
|
6772130f2a | ||
|
|
ddd72f3fac | ||
|
|
6e262835ef | ||
|
|
2f3b8ddc5f | ||
|
|
cea3a74b34 | ||
|
|
867941a233 | ||
|
|
3ff388a16d | ||
|
|
f4248e9ab9 | ||
|
|
507b3289c7 | ||
|
|
9e1dfc48d5 | ||
|
|
518cbc7b5d | ||
|
|
ccc8db0620 | ||
|
|
7cfb663efd | ||
|
|
e5103cc925 | ||
|
|
26458f5a19 | ||
|
|
79d5ec6caf | ||
|
|
034d461ab6 | ||
|
|
2e9c1c170c | ||
|
|
24ad3b2c61 | ||
|
|
288f55dc2f | ||
|
|
78dbea6267 | ||
|
|
6a9e53141d | ||
|
|
05e6994520 | ||
|
|
1a4dc67eb9 | ||
|
|
31d87a116b | ||
|
|
c47796d590 | ||
|
|
c7138a41ee | ||
|
|
96f04c70a9 | ||
|
|
87a8bc09ab | ||
|
|
5f5661d559 | ||
|
|
35ca87790e | ||
|
|
ae43e4a57c | ||
|
|
b91712a01a | ||
|
|
b20007b341 | ||
|
|
6a24e1188b | ||
|
|
5894efc1aa | ||
|
|
a05612d243 | ||
|
|
48de874d6b | ||
|
|
91e6da316f | ||
|
|
fa60bd81a1 | ||
|
|
a08a69c5be | ||
|
|
18d450a41a | ||
|
|
36cdee61c0 | ||
|
|
797e188259 | ||
|
|
91b581668a | ||
|
|
ad01fec28f | ||
|
|
357d3a0df6 | ||
|
|
5ce6022761 | ||
|
|
235a0c5aea | ||
|
|
9b81fa6ebb | ||
|
|
8792d5ab0e | ||
|
|
d46ed0c078 | ||
|
|
73c433fcd2 | ||
|
|
02b5239220 | ||
|
|
0ed3bdfcb0 | ||
|
|
bdeb89011f | ||
|
|
1523b625bc | ||
|
|
fb91eeb692 | ||
|
|
601d2e02cb | ||
|
|
0a662d34eb | ||
|
|
5cd4693e9d | ||
|
|
f3f0f860e3 | ||
|
|
93a5cf8a79 | ||
|
|
7cf15cbc21 | ||
|
|
30bc6d20b2 | ||
|
|
b39f99fde4 | ||
|
|
0e8aa9efa4 | ||
|
|
e1fcde36e3 | ||
|
|
7aafe077d3 | ||
|
|
5b8cab5e76 | ||
|
|
4ab56af40a | ||
|
|
e8cea86a76 | ||
|
|
d0a6e0b358 | ||
|
|
8831b3e970 | ||
|
|
f6db6f0914 | ||
|
|
ab8baeedd1 | ||
|
|
eccee5e72e | ||
|
|
4d93055bda | ||
|
|
c60c16e56a | ||
|
|
99b1de5647 | ||
|
|
7efe11a421 | ||
|
|
954108856a | ||
|
|
cbca745ec4 | ||
|
|
e15e7c8f8d | ||
|
|
65e8a520e5 | ||
|
|
3926af5a6d | ||
|
|
556fb33223 | ||
|
|
82295adeab | ||
|
|
efabf060c7 | ||
|
|
96e434ebf5 | ||
|
|
d81e2567cc | ||
|
|
f8d487639f | ||
|
|
cecfbb5375 | ||
|
|
8d35500e2f | ||
|
|
5dad5d8e03 | ||
|
|
9d7d4fad2e | ||
|
|
7be64bde02 | ||
|
|
48eb7552a9 | ||
|
|
5347b0060c | ||
|
|
b826c03226 | ||
|
|
1c211a8667 | ||
|
|
fd4d5b90e2 | ||
|
|
1ee9f26b34 | ||
|
|
da1762934b | ||
|
|
511457c761 | ||
|
|
29b4cedb81 | ||
|
|
585de15e6b | ||
|
|
74f7ad155d | ||
|
|
a9bf118f3a | ||
|
|
6d5a432bad | ||
|
|
f1f12abd16 | ||
|
|
09880a54e9 | ||
|
|
0f6847b16d | ||
|
|
ce82ed97f5 | ||
|
|
36b393dbde | ||
|
|
524c283a0d | ||
|
|
afda53a9bc | ||
|
|
1310d75012 | ||
|
|
80bbde549d | ||
|
|
2451487593 | ||
|
|
ecd626f105 | ||
|
|
123b312965 | ||
|
|
e94de8e629 | ||
|
|
956a5a04ca | ||
|
|
affeb7c624 | ||
|
|
e457d94df8 | ||
|
|
e9583c928e | ||
|
|
89c14628e1 | ||
|
|
ffba407eaf | ||
|
|
33f710127c | ||
|
|
7a82b2c102 | ||
|
|
2db2a47186 | ||
|
|
63faa43c1d | ||
|
|
eabb0e8470 | ||
|
|
9f75ae6b03 | ||
|
|
a1f28cd245 | ||
|
|
90a04b517e | ||
|
|
9f6e6a333f | ||
|
|
7b9e2c4fd9 | ||
|
|
63f13711cc | ||
|
|
c65b8e5ebd | ||
|
|
bfed1480ae | ||
|
|
5ff902f185 | ||
|
|
8d38345c7f | ||
|
|
14be9dbb8a | ||
|
|
720c26db94 | ||
|
|
6d0b8b845d | ||
|
|
b2767e000e | ||
|
|
169f23c2ca | ||
|
|
81f70eafff | ||
|
|
650170498a | ||
|
|
8b6f600989 | ||
|
|
fe3617b39f | ||
|
|
0f466c51ba | ||
|
|
a1a641bce3 | ||
|
|
4764c25eb1 | ||
|
|
d390455cf2 | ||
|
|
f58ebad0ec | ||
|
|
7ca4eb3b8f | ||
|
|
7fb8d24d73 | ||
|
|
472dbaa68b | ||
|
|
f03448007d | ||
|
|
c317a8bff9 | ||
|
|
618cca39a4 | ||
|
|
fe7a98098f | ||
|
|
df49939990 | ||
|
|
f23f2776f4 | ||
|
|
4419c86164 | ||
|
|
9848f49b49 | ||
|
|
679bd782a8 | ||
|
|
6a316e3906 | ||
|
|
c129db8474 | ||
|
|
10035b4c91 | ||
|
|
5839271de7 | ||
|
|
47db8ef709 | ||
|
|
2656491aaa | ||
|
|
a7637c9cae | ||
|
|
7b83ed8205 | ||
|
|
00cbc77f1d | ||
|
|
4d75b256c4 | ||
|
|
5aeff7c40c | ||
|
|
6a543bf644 | ||
|
|
dfb035525d | ||
|
|
4c23069a0a | ||
|
|
4a1d7be44c | ||
|
|
798aec1b74 | ||
|
|
7914d7e151 | ||
|
|
26d0392da1 | ||
|
|
83b1406cce | ||
|
|
8b579d6837 | ||
|
|
c0fd20dfff | ||
|
|
dd6b67c6e6 | ||
|
|
fa83185cf5 | ||
|
|
97d5010d41 | ||
|
|
e73ad93920 | ||
|
|
425af246fb | ||
|
|
a2912ba0ff | ||
|
|
48ff2f4413 | ||
|
|
61cd281a18 | ||
|
|
6e28134282 | ||
|
|
d1377f44d2 | ||
|
|
d261c6186b | ||
|
|
2a72a2612d | ||
|
|
66bb4f87d4 | ||
|
|
977b1ad083 | ||
|
|
94db18d42b | ||
|
|
7e0375108d | ||
|
|
094a5d7b62 | ||
|
|
8f947a5f04 | ||
|
|
5580ad62f9 | ||
|
|
c0953dc954 | ||
|
|
1df93da564 | ||
|
|
e2252a9d72 | ||
|
|
7cdba4b507 | ||
|
|
b110d5afec | ||
|
|
6112b3e399 | ||
|
|
af0869a39b | ||
|
|
d44c339990 | ||
|
|
0304f92ad9 | ||
|
|
4a41d6e5d5 | ||
|
|
30893ca68e | ||
|
|
1edd02fa5c | ||
|
|
10a957ff0d | ||
|
|
35b1b3619d | ||
|
|
777ebd0c41 | ||
|
|
9e7d8c08e1 | ||
|
|
ad3f4cda09 | ||
|
|
98502cc6ae | ||
|
|
0bac671eb0 | ||
|
|
09c9a1e752 | ||
|
|
41e77e8336 | ||
|
|
a6e9fee2a0 | ||
|
|
c403a1cee5 | ||
|
|
02d25b370a | ||
|
|
e5a3eec8a1 | ||
|
|
910352d66c | ||
|
|
c4f02e7d55 | ||
|
|
6a1197ad83 | ||
|
|
84c31bbb88 | ||
|
|
33f6c32306 | ||
|
|
5c71304d41 | ||
|
|
bbaf04e977 | ||
|
|
ad5b2d2eb3 | ||
|
|
3c064baf8a | ||
|
|
9e11b10d74 | ||
|
|
2fc45e00b4 | ||
|
|
45eb08fc3a | ||
|
|
fbcfae0200 | ||
|
|
9c4ce013ec | ||
|
|
5dba7c12f0 | ||
|
|
db793c80c8 | ||
|
|
aa4a3f1aa9 | ||
|
|
3424639309 | ||
|
|
0f309a29ba | ||
|
|
e13b8846b8 | ||
|
|
d5ea4d6129 | ||
|
|
9d50bfedbd | ||
|
|
b03410083c | ||
|
|
a314b1e448 | ||
|
|
e8a51a85c4 | ||
|
|
d4074f966c | ||
|
|
1413b41824 | ||
|
|
379c1edec3 | ||
|
|
58b5bade9e | ||
|
|
71aee4cd3e | ||
|
|
10bab0cfa1 | ||
|
|
698350b0f7 | ||
|
|
a97cf25031 | ||
|
|
8302d088bd | ||
|
|
64788e73de | ||
|
|
114fd471e8 | ||
|
|
b24a3120d3 | ||
|
|
c5f93b3d0a | ||
|
|
79290e4721 | ||
|
|
984e126f23 | ||
|
|
cd1ad31aed | ||
|
|
1e3f6cf9e7 | ||
|
|
9c6ccd9a8a | ||
|
|
681ba21d39 | ||
|
|
aef62189ee | ||
|
|
09f70ac817 | ||
|
|
1eacb22143 | ||
|
|
8524bad377 | ||
|
|
9d899243d1 | ||
|
|
9acf20a639 | ||
|
|
71ef6b2e82 | ||
|
|
5e34d95dd2 | ||
|
|
25a809c064 | ||
|
|
f760498150 | ||
|
|
328301a375 | ||
|
|
f28e7ebbbb | ||
|
|
bfa84af71e | ||
|
|
42cd1ba976 | ||
|
|
88cba866fd | ||
|
|
af2876a84b | ||
|
|
c5b15bfa78 | ||
|
|
85f00a63c2 | ||
|
|
05f427bcd7 | ||
|
|
69f4c41534 | ||
|
|
30b9239a8b | ||
|
|
2061a83c59 | ||
|
|
24e723de07 | ||
|
|
27b5694885 | ||
|
|
4093f28cee | ||
|
|
08aaf2f2dd | ||
|
|
5a927e5ba5 | ||
|
|
10fafcf59f | ||
|
|
be87591801 | ||
|
|
086d4378d3 | ||
|
|
e982275905 | ||
|
|
77c02703e9 | ||
|
|
0ef27d4f95 | ||
|
|
5edc27744e | ||
|
|
02ff887fee | ||
|
|
3a30eeb59f | ||
|
|
79af8fb601 | ||
|
|
f9f00faa0e | ||
|
|
a3c41e84e4 | ||
|
|
7f21f33351 | ||
|
|
568f682cee | ||
|
|
901c4c8ca4 | ||
|
|
3dbba97f9c | ||
|
|
3475620267 | ||
|
|
8936a5e5d8 | ||
|
|
719e34f9bc | ||
|
|
df955ff8b0 | ||
|
|
4dc74022b4 | ||
|
|
c2b7d07208 | ||
|
|
21e76a6f05 | ||
|
|
89c0d6fad6 | ||
|
|
b74d15b1ee | ||
|
|
10313438bf | ||
|
|
a764217948 | ||
|
|
d81d6fb937 | ||
|
|
4c0f7ab7f9 | ||
|
|
1c48945a96 | ||
|
|
850901f62f | ||
|
|
4822854e30 | ||
|
|
f580538331 | ||
|
|
0d70c555e6 | ||
|
|
c5f6cf0080 | ||
|
|
6ff7b3bc9a | ||
|
|
346c2c65b0 | ||
|
|
a6445fd500 | ||
|
|
358e43e508 | ||
|
|
20c5ba6b63 | ||
|
|
661658a6e8 | ||
|
|
0a6f224ed3 | ||
|
|
e7bb29693f | ||
|
|
32dfd50461 | ||
|
|
bfec385dce | ||
|
|
04278f99d7 | ||
|
|
9d3db19dc1 | ||
|
|
3953e33f37 | ||
|
|
611fdd56d0 | ||
|
|
36a4fd0f35 | ||
|
|
488984a988 | ||
|
|
1b183d6e58 | ||
|
|
3dcb6d30b5 | ||
|
|
1d11788f89 | ||
|
|
d3822128ab | ||
|
|
6ac7836505 | ||
|
|
5796fccae4 | ||
|
|
22ab6e544c | ||
|
|
5e735ab8bd | ||
|
|
450139402d | ||
|
|
1e0d4f1fde | ||
|
|
7e9b1af3a3 | ||
|
|
30b90cd4be | ||
|
|
f28affa222 | ||
|
|
2ae7b187b8 | ||
|
|
ad037be1f4 | ||
|
|
9e31542c5f | ||
|
|
aae6ed0cb3 | ||
|
|
7ea2ecb8e9 | ||
|
|
852fc28896 | ||
|
|
fcdab79802 | ||
|
|
bb0e06589f | ||
|
|
d65b72c4dd | ||
|
|
24a3d5b3de | ||
|
|
2814b46941 | ||
|
|
6ffc25448d | ||
|
|
d9a82f7c9f | ||
|
|
9a8106947e | ||
|
|
e21ca79ea1 | ||
|
|
dd579caeac | ||
|
|
ddf8a4b9fb | ||
|
|
3c04f4194e | ||
|
|
1e0de841eb | ||
|
|
c9c88cf0a6 | ||
|
|
fd1e124166 | ||
|
|
aa2bee258a | ||
|
|
12d696dc07 | ||
|
|
b33772a1d6 | ||
|
|
b41f89d7de | ||
|
|
e8ca213b74 | ||
|
|
7620b8d493 | ||
|
|
23306fd9e3 | ||
|
|
a68ccdca50 | ||
|
|
771e432546 | ||
|
|
20bdbfb81a | ||
|
|
90729b436c | ||
|
|
8ea4e97bbf | ||
|
|
17e12e9263 | ||
|
|
4cb3d021f4 | ||
|
|
97cec2e127 | ||
|
|
9c4294659c | ||
|
|
5b8526db96 | ||
|
|
54cce9e9fb | ||
|
|
54d1691d19 | ||
|
|
4d4431868e | ||
|
|
cdae69d346 | ||
|
|
3e86baabd8 | ||
|
|
6ec92b19fb | ||
|
|
630caa8686 | ||
|
|
497c3b2165 | ||
|
|
8823d06893 | ||
|
|
72a2bf50df | ||
|
|
15c1ed2011 | ||
|
|
3fb19663cc | ||
|
|
00cdbbc89c | ||
|
|
07f3cec16d | ||
|
|
e893e25ff4 | ||
|
|
1ad9a1eb13 | ||
|
|
d9288596e8 | ||
|
|
a2a642d9ce | ||
|
|
4f00007108 | ||
|
|
816ee77b6f | ||
|
|
26d470ecda | ||
|
|
3babd9d95e | ||
|
|
802e81b1cd | ||
|
|
41f0834c51 | ||
|
|
880de0b047 | ||
|
|
bbe7fda4e0 | ||
|
|
4fd214e328 | ||
|
|
92a9274dce | ||
|
|
8765d83083 | ||
|
|
a298152bc8 | ||
|
|
2b7abe5774 | ||
|
|
3e9241892e | ||
|
|
a8dcdcf339 | ||
|
|
6ea79a7960 | ||
|
|
2af562e313 | ||
|
|
064a36fcbb | ||
|
|
40672f6a47 | ||
|
|
6d66ae3f50 | ||
|
|
94c89e0325 | ||
|
|
3670ef40a3 | ||
|
|
16d464fba5 | ||
|
|
0b7e0b9cd0 | ||
|
|
dd5fd114d2 | ||
|
|
6e53879344 | ||
|
|
af2bfd4d59 | ||
|
|
a4a34c8ba7 | ||
|
|
8c609f4fdf | ||
|
|
197f5b583f | ||
|
|
4eb4a03e59 | ||
|
|
5719469452 | ||
|
|
6e9d525890 | ||
|
|
e82805dd48 | ||
|
|
3d4e66d438 | ||
|
|
ffbc9f169a | ||
|
|
99650187e9 | ||
|
|
92699317cd | ||
|
|
0e48315803 | ||
|
|
868ec246bd | ||
|
|
0030a688c6 | ||
|
|
3ba8f51a2f | ||
|
|
04576b335c | ||
|
|
ea29aa176f | ||
|
|
9a9b7289ba | ||
|
|
5edcd5572a | ||
|
|
d601790864 | ||
|
|
3cd9b819f5 | ||
|
|
bf58d3ae93 | ||
|
|
d6c32e2d39 | ||
|
|
6914f75bb7 | ||
|
|
3adf3946b5 | ||
|
|
bdc4948afb | ||
|
|
331db10029 | ||
|
|
c036a157c8 | ||
|
|
2daf9e34d2 | ||
|
|
acf35f8c51 | ||
|
|
9155515082 | ||
|
|
558cd9b8b3 | ||
|
|
a0f02d0d2f | ||
|
|
9feb8492aa | ||
|
|
e5aa726899 | ||
|
|
93d1c28ccd | ||
|
|
b5ba9200bc | ||
|
|
699d866670 | ||
|
|
c3071344cb | ||
|
|
9e998dd2b6 | ||
|
|
c9accf6079 | ||
|
|
1b41a51004 | ||
|
|
3338dce900 | ||
|
|
1364779f81 | ||
|
|
b49d3423fc | ||
|
|
cccab2a985 | ||
|
|
1abaa90a7d | ||
|
|
6e1993ca8a | ||
|
|
171c821ac4 | ||
|
|
7ebf9186bf | ||
|
|
57d2f2baef | ||
|
|
0aee13878a | ||
|
|
f93ef0ca76 | ||
|
|
4ec03d8338 | ||
|
|
cb318aa6c6 | ||
|
|
733482cd5c | ||
|
|
3969d1287d | ||
|
|
1aa7854b0d | ||
|
|
7b27d4a1a7 | ||
|
|
24ddb8771f | ||
|
|
7134714245 | ||
|
|
96b320ac26 | ||
|
|
61235828ce | ||
|
|
1a27b21efe | ||
|
|
b88e121b30 | ||
|
|
4ba4119c2b | ||
|
|
91d1ca201c | ||
|
|
8bf063a228 | ||
|
|
4f53de39b1 | ||
|
|
8c3804f107 | ||
|
|
1be4ec2b90 | ||
|
|
8f0ed44b11 | ||
|
|
cbadfc4ac4 | ||
|
|
3d88ae4452 | ||
|
|
e57f565812 | ||
|
|
da2605ee03 | ||
|
|
07e2196eb4 | ||
|
|
6d99c54af7 | ||
|
|
2b7901e9a8 | ||
|
|
fb06dd1dbc | ||
|
|
d3b825529e | ||
|
|
ccf9c0db22 | ||
|
|
f8ba36b8be | ||
|
|
927c07bfd5 | ||
|
|
5bf9d99b3d | ||
|
|
7cad05342a | ||
|
|
867780e525 | ||
|
|
ff4f9a79c9 | ||
|
|
6699c36fb3 | ||
|
|
abd4556994 | ||
|
|
ccf0d17371 | ||
|
|
898584bbb6 | ||
|
|
6d7a267e39 | ||
|
|
9f656ca3cb | ||
|
|
d00caf1f4c | ||
|
|
a5e9d72bc5 | ||
|
|
8ac9047831 | ||
|
|
fede6451e2 | ||
|
|
9797ad380c | ||
|
|
33bc4a4d22 | ||
|
|
65509ace59 | ||
|
|
ea173f971a | ||
|
|
6378754c57 | ||
|
|
30fc972d78 | ||
|
|
c022b31c79 | ||
|
|
049b06bb39 | ||
|
|
e17d5213c0 | ||
|
|
dcf681941e | ||
|
|
1cd7d40405 | ||
|
|
fbd80ba2c7 | ||
|
|
88ab85bd04 | ||
|
|
78f98744fd | ||
|
|
9c9634a927 | ||
|
|
be47be626c | ||
|
|
2fbd3d8e19 | ||
|
|
d5c3d4c051 | ||
|
|
fac60f7ddd | ||
|
|
c371478c61 | ||
|
|
5911e29f34 | ||
|
|
c45c97c5d8 | ||
|
|
99d68dfc0e | ||
|
|
c9b366f3e2 | ||
|
|
4e732e9491 | ||
|
|
dd5b12aa38 | ||
|
|
7bd960fba9 | ||
|
|
c338c33902 | ||
|
|
df6b7ae635 | ||
|
|
a3346f8223 | ||
|
|
196f2c2c3b | ||
|
|
77d549ac1b | ||
|
|
5c3cce66c1 | ||
|
|
cc2f09601e | ||
|
|
14af04cc73 | ||
|
|
37c9e68c21 | ||
|
|
2bd343b2da | ||
|
|
f5d502c5ad | ||
|
|
35cf460ccf | ||
|
|
28c7b90c3f | ||
|
|
4fbdaf42e1 | ||
|
|
90910620d9 | ||
|
|
eb4336fef7 | ||
|
|
69264cc8ec | ||
|
|
ab0cb74ca9 | ||
|
|
42101ab6fd | ||
|
|
8d69c70076 | ||
|
|
beb3077159 | ||
|
|
ecb3ca2b4e | ||
|
|
2ba42e0c25 | ||
|
|
3ef5590e18 | ||
|
|
8412e3867d | ||
|
|
90c40100d1 | ||
|
|
92cb49da90 | ||
|
|
abc09c067f | ||
|
|
edbd1e4bbc | ||
|
|
75edb91825 | ||
|
|
602a61b08d | ||
|
|
d8222d83f0 | ||
|
|
7da5512d45 | ||
|
|
8bf9ae7824 | ||
|
|
f57777e417 | ||
|
|
b3cc3d857a | ||
|
|
bf442d9e70 | ||
|
|
1a556d05ba | ||
|
|
dab301e6d3 | ||
|
|
8ab4b4c788 | ||
|
|
4b29060c96 | ||
|
|
8a5f96a847 | ||
|
|
149fa57075 | ||
|
|
affe184ccd | ||
|
|
1e5e73c4ae | ||
|
|
c76316da03 | ||
|
|
de6205f860 | ||
|
|
f994255091 | ||
|
|
6d4981a3db | ||
|
|
06fef2296f | ||
|
|
999a702291 | ||
|
|
020b9bb2c2 | ||
|
|
7713caab51 | ||
|
|
97a777d729 | ||
|
|
8241d1f08c | ||
|
|
2ac85bbfff | ||
|
|
3f68ae2235 | ||
|
|
0f7b6f75df | ||
|
|
b048e8f5ca | ||
|
|
9783dc45ff | ||
|
|
badbefbade | ||
|
|
b6a265cfad | ||
|
|
9b5ea2f84b | ||
|
|
d8acfa5c05 | ||
|
|
2a5b6924eb | ||
|
|
3b9aee1eb9 | ||
|
|
83801c9063 | ||
|
|
0f34350c58 | ||
|
|
1c4c0f0715 | ||
|
|
3825fc831a | ||
|
|
654670cbc8 | ||
|
|
947570072c | ||
|
|
01b6b45380 | ||
|
|
b9dc1f8357 | ||
|
|
b4bd810c9d | ||
|
|
490b037920 | ||
|
|
cdff5458bc | ||
|
|
09314a09e9 | ||
|
|
49ea8edb27 | ||
|
|
86e5ff2fff | ||
|
|
1ade0ae6b9 | ||
|
|
cadfb59b8d | ||
|
|
5c2013f19c | ||
|
|
8db46113f4 | ||
|
|
3028a0b1c0 | ||
|
|
e9b4e39061 | ||
|
|
9c6715e4e5 | ||
|
|
0b03d5ee90 | ||
|
|
30fecf6428 | ||
|
|
752eaaa0f5 | ||
|
|
006d10381f | ||
|
|
a1ffe3cc0e | ||
|
|
b4a1cbbf55 | ||
|
|
b2ae5a33ae | ||
|
|
aec0a75e87 | ||
|
|
7086413f0c | ||
|
|
d55f03b63c | ||
|
|
aa4fd2fe90 | ||
|
|
6abf628a38 | ||
|
|
ad46002c85 | ||
|
|
2f21bd0f44 | ||
|
|
993608f911 | ||
|
|
c6c6adb7d8 | ||
|
|
3937330ce4 | ||
|
|
1590c848c9 | ||
|
|
2bb45b312c | ||
|
|
1fc95c96eb | ||
|
|
ee7a453a72 | ||
|
|
4b79afbac0 | ||
|
|
c8fc31257b | ||
|
|
8e0b8fd7f9 | ||
|
|
ee8f9e4d24 | ||
|
|
994e03945d | ||
|
|
aff00a18b5 | ||
|
|
6c22e6554d | ||
|
|
2a0d7654e7 | ||
|
|
4eb1f641ae | ||
|
|
2da5a243ec | ||
|
|
5ac8ccbe5c | ||
|
|
0568533550 | ||
|
|
178abc2af2 | ||
|
|
adb2a5f459 | ||
|
|
ada1571e1e | ||
|
|
5931c00ff3 | ||
|
|
a6e7c1bf74 | ||
|
|
1a5374f2f6 | ||
|
|
c9e3683b8e | ||
|
|
aba93b342a | ||
|
|
dee78b77a9 | ||
|
|
d21705f355 | ||
|
|
9abcd4bd0b | ||
|
|
b052943e34 | ||
|
|
e1e9b4c2e8 | ||
|
|
42c30e0741 | ||
|
|
3b45e77e65 | ||
|
|
dcb2b6b912 | ||
|
|
638a4e2535 | ||
|
|
489fde16d1 | ||
|
|
35e1c363e5 |
13
.check.exs
13
.check.exs
@@ -13,8 +13,8 @@
|
|||||||
|
|
||||||
## list of tools (see `mix check` docs for a list of default curated tools)
|
## list of tools (see `mix check` docs for a list of default curated tools)
|
||||||
tools: [
|
tools: [
|
||||||
## curated tools may be disabled (e.g. the check for compilation warnings)
|
## Allow compilation warnings for now (error budget: unlimited warnings)
|
||||||
{:compiler, false},
|
{:compiler, "mix compile"},
|
||||||
|
|
||||||
## ...or have command & args adjusted (e.g. enable skip comments for sobelow)
|
## ...or have command & args adjusted (e.g. enable skip comments for sobelow)
|
||||||
# {:sobelow, "mix sobelow --exit --skip"},
|
# {:sobelow, "mix sobelow --exit --skip"},
|
||||||
@@ -22,10 +22,15 @@
|
|||||||
## ...or reordered (e.g. to see output from dialyzer before others)
|
## ...or reordered (e.g. to see output from dialyzer before others)
|
||||||
# {:dialyzer, order: -1},
|
# {:dialyzer, order: -1},
|
||||||
|
|
||||||
## ...or reconfigured (e.g. disable parallel execution of ex_unit in umbrella)
|
## Credo with relaxed error budget: max 200 issues
|
||||||
|
{:credo, "mix credo --strict --max-issues 200"},
|
||||||
|
|
||||||
|
## Dialyzer but don't halt on exit (allow warnings)
|
||||||
|
{:dialyzer, "mix dialyzer"},
|
||||||
|
|
||||||
|
## Tests without warnings-as-errors for now
|
||||||
|
{:ex_unit, "mix test"},
|
||||||
{:doctor, false},
|
{:doctor, false},
|
||||||
{:ex_unit, false},
|
|
||||||
{:npm_test, false},
|
{:npm_test, false},
|
||||||
{:sobelow, false}
|
{:sobelow, false}
|
||||||
|
|
||||||
|
|||||||
18
.credo.exs
18
.credo.exs
@@ -82,8 +82,6 @@
|
|||||||
# You can customize the priority of any check
|
# You can customize the priority of any check
|
||||||
# Priority values are: `low, normal, high, higher`
|
# Priority values are: `low, normal, high, higher`
|
||||||
#
|
#
|
||||||
{Credo.Check.Design.AliasUsage,
|
|
||||||
[priority: :low, if_nested_deeper_than: 2, if_called_more_often_than: 0]},
|
|
||||||
# You can also customize the exit_status of each check.
|
# You can also customize the exit_status of each check.
|
||||||
# If you don't want TODO comments to cause `mix credo` to fail, just
|
# If you don't want TODO comments to cause `mix credo` to fail, just
|
||||||
# set this value to 0 (zero).
|
# set this value to 0 (zero).
|
||||||
@@ -99,10 +97,9 @@
|
|||||||
{Credo.Check.Readability.LargeNumbers, []},
|
{Credo.Check.Readability.LargeNumbers, []},
|
||||||
{Credo.Check.Readability.MaxLineLength, [priority: :low, max_length: 120]},
|
{Credo.Check.Readability.MaxLineLength, [priority: :low, max_length: 120]},
|
||||||
{Credo.Check.Readability.ModuleAttributeNames, []},
|
{Credo.Check.Readability.ModuleAttributeNames, []},
|
||||||
{Credo.Check.Readability.ModuleDoc, []},
|
{Credo.Check.Readability.ModuleDoc, false},
|
||||||
{Credo.Check.Readability.ModuleNames, []},
|
{Credo.Check.Readability.ModuleNames, []},
|
||||||
{Credo.Check.Readability.ParenthesesInCondition, []},
|
{Credo.Check.Readability.ParenthesesInCondition, []},
|
||||||
{Credo.Check.Readability.ParenthesesOnZeroArityDefs, []},
|
|
||||||
{Credo.Check.Readability.PipeIntoAnonymousFunctions, []},
|
{Credo.Check.Readability.PipeIntoAnonymousFunctions, []},
|
||||||
{Credo.Check.Readability.PredicateFunctionNames, []},
|
{Credo.Check.Readability.PredicateFunctionNames, []},
|
||||||
{Credo.Check.Readability.PreferImplicitTry, []},
|
{Credo.Check.Readability.PreferImplicitTry, []},
|
||||||
@@ -121,14 +118,12 @@
|
|||||||
#
|
#
|
||||||
{Credo.Check.Refactor.Apply, []},
|
{Credo.Check.Refactor.Apply, []},
|
||||||
{Credo.Check.Refactor.CondStatements, []},
|
{Credo.Check.Refactor.CondStatements, []},
|
||||||
{Credo.Check.Refactor.CyclomaticComplexity, []},
|
|
||||||
{Credo.Check.Refactor.FunctionArity, []},
|
{Credo.Check.Refactor.FunctionArity, []},
|
||||||
{Credo.Check.Refactor.LongQuoteBlocks, []},
|
{Credo.Check.Refactor.LongQuoteBlocks, []},
|
||||||
{Credo.Check.Refactor.MatchInCondition, []},
|
{Credo.Check.Refactor.MatchInCondition, []},
|
||||||
{Credo.Check.Refactor.MapJoin, []},
|
{Credo.Check.Refactor.MapJoin, []},
|
||||||
{Credo.Check.Refactor.NegatedConditionsInUnless, []},
|
{Credo.Check.Refactor.NegatedConditionsInUnless, []},
|
||||||
{Credo.Check.Refactor.NegatedConditionsWithElse, []},
|
{Credo.Check.Refactor.NegatedConditionsWithElse, []},
|
||||||
{Credo.Check.Refactor.Nesting, []},
|
|
||||||
{Credo.Check.Refactor.UnlessWithElse, []},
|
{Credo.Check.Refactor.UnlessWithElse, []},
|
||||||
{Credo.Check.Refactor.WithClauses, []},
|
{Credo.Check.Refactor.WithClauses, []},
|
||||||
{Credo.Check.Refactor.FilterFilter, []},
|
{Credo.Check.Refactor.FilterFilter, []},
|
||||||
@@ -196,10 +191,19 @@
|
|||||||
{Credo.Check.Warning.LeakyEnvironment, []},
|
{Credo.Check.Warning.LeakyEnvironment, []},
|
||||||
{Credo.Check.Warning.MapGetUnsafePass, []},
|
{Credo.Check.Warning.MapGetUnsafePass, []},
|
||||||
{Credo.Check.Warning.MixEnv, []},
|
{Credo.Check.Warning.MixEnv, []},
|
||||||
{Credo.Check.Warning.UnsafeToAtom, []}
|
{Credo.Check.Warning.UnsafeToAtom, []},
|
||||||
|
|
||||||
# {Credo.Check.Refactor.MapInto, []},
|
# {Credo.Check.Refactor.MapInto, []},
|
||||||
|
|
||||||
|
#
|
||||||
|
# Temporarily disable checks that generate too many issues
|
||||||
|
# to get under the 200 issue budget
|
||||||
|
#
|
||||||
|
{Credo.Check.Readability.ParenthesesOnZeroArityDefs, []},
|
||||||
|
{Credo.Check.Design.AliasUsage, []},
|
||||||
|
{Credo.Check.Refactor.Nesting, []},
|
||||||
|
{Credo.Check.Refactor.CyclomaticComplexity, []}
|
||||||
|
|
||||||
#
|
#
|
||||||
# Custom checks can be created using `mix credo.gen.check`.
|
# Custom checks can be created using `mix credo.gen.check`.
|
||||||
#
|
#
|
||||||
|
|||||||
127
.credo.test.exs
Normal file
127
.credo.test.exs
Normal file
@@ -0,0 +1,127 @@
|
|||||||
|
# Credo configuration specific to test files
|
||||||
|
# This enforces stricter quality standards for test code
|
||||||
|
|
||||||
|
%{
|
||||||
|
configs: [
|
||||||
|
%{
|
||||||
|
name: "test",
|
||||||
|
files: %{
|
||||||
|
included: ["test/"],
|
||||||
|
excluded: ["test/support/"]
|
||||||
|
},
|
||||||
|
requires: [],
|
||||||
|
strict: true,
|
||||||
|
color: true,
|
||||||
|
checks: [
|
||||||
|
# Consistency checks
|
||||||
|
{Credo.Check.Consistency.ExceptionNames, []},
|
||||||
|
{Credo.Check.Consistency.LineEndings, []},
|
||||||
|
{Credo.Check.Consistency.MultiAliasImportRequireUse, []},
|
||||||
|
{Credo.Check.Consistency.ParameterPatternMatching, []},
|
||||||
|
{Credo.Check.Consistency.SpaceAroundOperators, []},
|
||||||
|
{Credo.Check.Consistency.SpaceInParentheses, []},
|
||||||
|
{Credo.Check.Consistency.TabsOrSpaces, []},
|
||||||
|
|
||||||
|
# Design checks - stricter for tests
|
||||||
|
{Credo.Check.Design.AliasUsage, priority: :high},
|
||||||
|
# Lower threshold for tests
|
||||||
|
{Credo.Check.Design.DuplicatedCode, mass_threshold: 25},
|
||||||
|
{Credo.Check.Design.TagTODO, []},
|
||||||
|
{Credo.Check.Design.TagFIXME, []},
|
||||||
|
|
||||||
|
# Readability checks - very important for tests
|
||||||
|
{Credo.Check.Readability.AliasOrder, []},
|
||||||
|
{Credo.Check.Readability.FunctionNames, []},
|
||||||
|
{Credo.Check.Readability.LargeNumbers, []},
|
||||||
|
# Slightly longer for test descriptions
|
||||||
|
{Credo.Check.Readability.MaxLineLength, max_length: 120},
|
||||||
|
{Credo.Check.Readability.ModuleAttributeNames, []},
|
||||||
|
# Not required for test modules
|
||||||
|
{Credo.Check.Readability.ModuleDoc, false},
|
||||||
|
{Credo.Check.Readability.ModuleNames, []},
|
||||||
|
{Credo.Check.Readability.ParenthesesInCondition, []},
|
||||||
|
{Credo.Check.Readability.ParenthesesOnZeroArityDefs, []},
|
||||||
|
{Credo.Check.Readability.PredicateFunctionNames, []},
|
||||||
|
{Credo.Check.Readability.PreferImplicitTry, []},
|
||||||
|
{Credo.Check.Readability.RedundantBlankLines, []},
|
||||||
|
{Credo.Check.Readability.Semicolons, []},
|
||||||
|
{Credo.Check.Readability.SpaceAfterCommas, []},
|
||||||
|
{Credo.Check.Readability.StringSigils, []},
|
||||||
|
{Credo.Check.Readability.TrailingBlankLine, []},
|
||||||
|
{Credo.Check.Readability.TrailingWhiteSpace, []},
|
||||||
|
{Credo.Check.Readability.UnnecessaryAliasExpansion, []},
|
||||||
|
{Credo.Check.Readability.VariableNames, []},
|
||||||
|
{Credo.Check.Readability.WithSingleClause, []},
|
||||||
|
|
||||||
|
# Test-specific readability checks
|
||||||
|
# Discourage single pipes in tests
|
||||||
|
{Credo.Check.Readability.SinglePipe, []},
|
||||||
|
# Specs not needed in tests
|
||||||
|
{Credo.Check.Readability.Specs, false},
|
||||||
|
{Credo.Check.Readability.StrictModuleLayout, []},
|
||||||
|
|
||||||
|
# Refactoring opportunities - important for test maintainability
|
||||||
|
# Higher limit for complex test setups
|
||||||
|
{Credo.Check.Refactor.ABCSize, max_size: 50},
|
||||||
|
{Credo.Check.Refactor.AppendSingleItem, []},
|
||||||
|
{Credo.Check.Refactor.CondStatements, []},
|
||||||
|
{Credo.Check.Refactor.CyclomaticComplexity, max_complexity: 10},
|
||||||
|
# Lower for test helpers
|
||||||
|
{Credo.Check.Refactor.FunctionArity, max_arity: 4},
|
||||||
|
{Credo.Check.Refactor.LongQuoteBlocks, []},
|
||||||
|
{Credo.Check.Refactor.MapInto, []},
|
||||||
|
{Credo.Check.Refactor.MatchInCondition, []},
|
||||||
|
{Credo.Check.Refactor.NegatedConditionsInUnless, []},
|
||||||
|
{Credo.Check.Refactor.NegatedConditionsWithElse, []},
|
||||||
|
# Keep tests flat
|
||||||
|
{Credo.Check.Refactor.Nesting, max_nesting: 3},
|
||||||
|
{Credo.Check.Refactor.UnlessWithElse, []},
|
||||||
|
{Credo.Check.Refactor.WithClauses, []},
|
||||||
|
{Credo.Check.Refactor.FilterFilter, []},
|
||||||
|
{Credo.Check.Refactor.RejectReject, []},
|
||||||
|
{Credo.Check.Refactor.RedundantWithClauseResult, []},
|
||||||
|
|
||||||
|
# Warnings - all should be fixed
|
||||||
|
{Credo.Check.Warning.ApplicationConfigInModuleAttribute, []},
|
||||||
|
{Credo.Check.Warning.BoolOperationOnSameValues, []},
|
||||||
|
{Credo.Check.Warning.ExpensiveEmptyEnumCheck, []},
|
||||||
|
{Credo.Check.Warning.IExPry, []},
|
||||||
|
{Credo.Check.Warning.IoInspect, []},
|
||||||
|
{Credo.Check.Warning.OperationOnSameValues, []},
|
||||||
|
{Credo.Check.Warning.OperationWithConstantResult, []},
|
||||||
|
{Credo.Check.Warning.RaiseInsideRescue, []},
|
||||||
|
{Credo.Check.Warning.UnusedEnumOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedFileOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedKeywordOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedListOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedPathOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedRegexOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedStringOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedTupleOperation, []},
|
||||||
|
{Credo.Check.Warning.UnsafeExec, []},
|
||||||
|
|
||||||
|
# Test-specific checks
|
||||||
|
# Important for test isolation
|
||||||
|
{Credo.Check.Warning.LeakyEnvironment, []},
|
||||||
|
|
||||||
|
# Custom checks for test patterns
|
||||||
|
{
|
||||||
|
Credo.Check.Refactor.PipeChainStart,
|
||||||
|
# Factory functions
|
||||||
|
excluded_functions: ["build", "create", "insert"],
|
||||||
|
excluded_argument_types: [:atom, :number]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
# Disable these checks for test files
|
||||||
|
disabled: [
|
||||||
|
# Tests don't need module docs
|
||||||
|
{Credo.Check.Readability.ModuleDoc, []},
|
||||||
|
# Tests don't need specs
|
||||||
|
{Credo.Check.Readability.Specs, []},
|
||||||
|
# Common in test setup
|
||||||
|
{Credo.Check.Refactor.VariableRebinding, []}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,6 +1,27 @@
|
|||||||
FROM elixir:1.17-otp-27
|
FROM elixir:1.17-otp-27
|
||||||
|
|
||||||
RUN apt install -yq curl gnupg
|
RUN apt install -yq curl gnupg
|
||||||
|
# Install OS packages and Node.js (via nodesource),
|
||||||
|
# plus inotify-tools and yarn
|
||||||
|
RUN apt-get update && apt-get install -y --no-install-recommends \
|
||||||
|
sudo \
|
||||||
|
curl \
|
||||||
|
make \
|
||||||
|
git \
|
||||||
|
bash \
|
||||||
|
build-essential \
|
||||||
|
ca-certificates \
|
||||||
|
jq \
|
||||||
|
vim \
|
||||||
|
net-tools \
|
||||||
|
procps \
|
||||||
|
# Optionally add any other tools you need, e.g. vim, wget...
|
||||||
|
&& curl -sL https://deb.nodesource.com/setup_18.x | bash - \
|
||||||
|
&& apt-get install -y --no-install-recommends nodejs inotify-tools \
|
||||||
|
&& npm install -g yarn \
|
||||||
|
&& apt-get clean \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
RUN apt --fix-broken install
|
RUN apt --fix-broken install
|
||||||
|
|
||||||
RUN mix local.hex --force
|
RUN mix local.hex --force
|
||||||
|
|||||||
@@ -1,20 +1,30 @@
|
|||||||
{
|
{
|
||||||
"name": "wanderer-dev",
|
"name": "wanderer-dev",
|
||||||
"dockerComposeFile": ["./docker-compose.yml"],
|
"dockerComposeFile": ["./docker-compose.yml"],
|
||||||
"extensions": [
|
"customizations": {
|
||||||
"jakebecker.elixir-ls",
|
"vscode": {
|
||||||
"JakeBecker.elixir-ls",
|
"extensions": [
|
||||||
"dbaeumer.vscode-eslint",
|
"jakebecker.elixir-ls",
|
||||||
"esbenp.prettier-vscode"
|
"JakeBecker.elixir-ls",
|
||||||
],
|
"dbaeumer.vscode-eslint",
|
||||||
|
"esbenp.prettier-vscode"
|
||||||
|
],
|
||||||
|
"settings": {
|
||||||
|
"editor.formatOnSave": true,
|
||||||
|
"search.exclude": {
|
||||||
|
"**/doc": true
|
||||||
|
},
|
||||||
|
"elixirLS.dialyzerEnabled": false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"service": "wanderer",
|
"service": "wanderer",
|
||||||
"workspaceFolder": "/app",
|
"workspaceFolder": "/app",
|
||||||
"shutdownAction": "stopCompose",
|
"shutdownAction": "stopCompose",
|
||||||
"settings": {
|
"features": {
|
||||||
"editor.formatOnSave": true,
|
"ghcr.io/devcontainers/features/common-utils:2": {
|
||||||
"search.exclude": {
|
"networkArgs": ["--add-host=host.docker.internal:host-gateway"]
|
||||||
"**/doc": true
|
}
|
||||||
},
|
},
|
||||||
"elixirLS.dialyzerEnabled": false
|
"forwardPorts": [4444]
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,15 +14,15 @@ services:
|
|||||||
|
|
||||||
wanderer:
|
wanderer:
|
||||||
environment:
|
environment:
|
||||||
PORT: 8000
|
PORT: 4444
|
||||||
DB_HOST: db
|
DB_HOST: db
|
||||||
WEB_APP_URL: "http://localhost:8000"
|
WEB_APP_URL: "http://localhost:4444"
|
||||||
ERL_AFLAGS: "-kernel shell_history enabled"
|
ERL_AFLAGS: "-kernel shell_history enabled"
|
||||||
build:
|
build:
|
||||||
context: .
|
context: .
|
||||||
dockerfile: Dockerfile
|
dockerfile: Dockerfile
|
||||||
ports:
|
ports:
|
||||||
- 8000:8000
|
- 4444:4444
|
||||||
volumes:
|
volumes:
|
||||||
- ..:/app:delegated
|
- ..:/app:delegated
|
||||||
- ~/.gitconfig:/root/.gitconfig
|
- ~/.gitconfig:/root/.gitconfig
|
||||||
|
|||||||
39
.devcontainer/setup.sh
Executable file
39
.devcontainer/setup.sh
Executable file
@@ -0,0 +1,39 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
echo "→ fetching & compiling deps"
|
||||||
|
mix deps.get
|
||||||
|
mix compile
|
||||||
|
|
||||||
|
# only run Ecto if the project actually has those tasks
|
||||||
|
if mix help | grep -q "ecto.create"; then
|
||||||
|
echo "→ waiting for database to be ready..."
|
||||||
|
|
||||||
|
# Wait for database to be ready
|
||||||
|
DB_HOST=${DB_HOST:-db}
|
||||||
|
timeout=60
|
||||||
|
while ! nc -z $DB_HOST 5432 2>/dev/null; do
|
||||||
|
if [ $timeout -eq 0 ]; then
|
||||||
|
echo "❌ Database connection timeout"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "Waiting for database... ($timeout seconds remaining)"
|
||||||
|
sleep 1
|
||||||
|
timeout=$((timeout - 1))
|
||||||
|
done
|
||||||
|
|
||||||
|
# Give the database a bit more time to fully initialize
|
||||||
|
echo "→ giving database 2 more seconds to fully initialize..."
|
||||||
|
sleep 2
|
||||||
|
|
||||||
|
echo "→ database is ready, running ecto.create && ecto.migrate"
|
||||||
|
mix ecto.create --quiet
|
||||||
|
mix ecto.migrate
|
||||||
|
fi
|
||||||
|
|
||||||
|
cd assets
|
||||||
|
echo "→ installing JS & CSS dependencies"
|
||||||
|
yarn install --frozen-lockfile
|
||||||
|
echo "→ building assets"
|
||||||
|
|
||||||
|
echo "✅ setup complete"
|
||||||
@@ -8,4 +8,9 @@ export GIT_SHA="1111"
|
|||||||
export WANDERER_INVITES="false"
|
export WANDERER_INVITES="false"
|
||||||
export WANDERER_PUBLIC_API_DISABLED="false"
|
export WANDERER_PUBLIC_API_DISABLED="false"
|
||||||
export WANDERER_CHARACTER_API_DISABLED="false"
|
export WANDERER_CHARACTER_API_DISABLED="false"
|
||||||
export WANDERER_ZKILL_PRELOAD_DISABLED="false"
|
export WANDERER_KILLS_SERVICE_ENABLED="true"
|
||||||
|
export WANDERER_KILLS_BASE_URL="ws://host.docker.internal:4004"
|
||||||
|
export WANDERER_SSE_ENABLED="true"
|
||||||
|
export WANDERER_WEBHOOKS_ENABLED="true"
|
||||||
|
export WANDERER_SSE_MAX_CONNECTIONS="1000"
|
||||||
|
export WANDERER_WEBHOOK_TIMEOUT_MS="15000"
|
||||||
109
.github/workflows/advanced-test.yml
vendored
Normal file
109
.github/workflows/advanced-test.yml
vendored
Normal file
@@ -0,0 +1,109 @@
|
|||||||
|
name: Build Test
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- develop
|
||||||
|
|
||||||
|
env:
|
||||||
|
MIX_ENV: prod
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
REGISTRY_IMAGE: wandererltd/community-edition
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: ${{ github.workflow }}-${{ github.ref }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy-test:
|
||||||
|
name: 🚀 Deploy to test env (fly.io)
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ github.base_ref == 'develop' || (github.ref == 'refs/heads/develop' && github.event_name == 'push') }}
|
||||||
|
steps:
|
||||||
|
- name: ⬇️ Checkout repo
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- uses: superfly/flyctl-actions/setup-flyctl@master
|
||||||
|
|
||||||
|
- name: 👀 Read app name
|
||||||
|
uses: SebRollen/toml-action@v1.0.0
|
||||||
|
id: app_name
|
||||||
|
with:
|
||||||
|
file: "fly.toml"
|
||||||
|
field: "app"
|
||||||
|
|
||||||
|
- name: 🚀 Deploy Test
|
||||||
|
run: flyctl deploy --remote-only --wait-timeout=300 --ha=false
|
||||||
|
env:
|
||||||
|
FLY_API_TOKEN: ${{ secrets.FLY_API_TOKEN }}
|
||||||
|
|
||||||
|
build:
|
||||||
|
name: 🛠 Build
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
if: ${{ (github.ref == 'refs/heads/develop') && github.event_name == 'push' }}
|
||||||
|
permissions:
|
||||||
|
checks: write
|
||||||
|
contents: write
|
||||||
|
packages: write
|
||||||
|
attestations: write
|
||||||
|
id-token: write
|
||||||
|
pull-requests: write
|
||||||
|
repository-projects: write
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
otp: ["27"]
|
||||||
|
elixir: ["1.17"]
|
||||||
|
node-version: ["18.x"]
|
||||||
|
outputs:
|
||||||
|
commit_hash: ${{ steps.generate-changelog.outputs.commit_hash }}
|
||||||
|
steps:
|
||||||
|
- name: Prepare
|
||||||
|
run: |
|
||||||
|
platform=${{ matrix.platform }}
|
||||||
|
echo "PLATFORM_PAIR=${platform//\//-}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Setup Elixir
|
||||||
|
uses: erlef/setup-beam@v1
|
||||||
|
with:
|
||||||
|
otp-version: ${{matrix.otp}}
|
||||||
|
elixir-version: ${{matrix.elixir}}
|
||||||
|
# nix build would also work here because `todos` is the default package
|
||||||
|
- name: ⬇️ Checkout repo
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
- name: 😅 Cache deps
|
||||||
|
id: cache-deps
|
||||||
|
uses: actions/cache@v4
|
||||||
|
env:
|
||||||
|
cache-name: cache-elixir-deps
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
deps
|
||||||
|
key: ${{ runner.os }}-mix-${{ matrix.elixir }}-${{ matrix.otp }}-${{ hashFiles('**/mix.lock') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-mix-${{ matrix.elixir }}-${{ matrix.otp }}-
|
||||||
|
- name: 😅 Cache compiled build
|
||||||
|
id: cache-build
|
||||||
|
uses: actions/cache@v4
|
||||||
|
env:
|
||||||
|
cache-name: cache-compiled-build
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
_build
|
||||||
|
key: ${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-${{ hashFiles( '**/lib/**/*.{ex,eex}', '**/config/*.exs', '**/mix.exs' ) }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-
|
||||||
|
${{ runner.os }}-build-
|
||||||
|
# Step: Download project dependencies. If unchanged, uses
|
||||||
|
# the cached version.
|
||||||
|
- name: 🌐 Install dependencies
|
||||||
|
run: mix deps.get --only "prod"
|
||||||
|
|
||||||
|
# Step: Compile the project treating any warnings as errors.
|
||||||
|
# Customize this step if a different behavior is desired.
|
||||||
|
- name: 🛠 Compiles without warnings
|
||||||
|
if: steps.cache-build.outputs.cache-hit != 'true'
|
||||||
|
run: mix compile
|
||||||
113
.github/workflows/build.yml
vendored
113
.github/workflows/build.yml
vendored
@@ -4,7 +4,8 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
- "releases/*"
|
- develop
|
||||||
|
|
||||||
env:
|
env:
|
||||||
MIX_ENV: prod
|
MIX_ENV: prod
|
||||||
GH_TOKEN: ${{ github.token }}
|
GH_TOKEN: ${{ github.token }}
|
||||||
@@ -18,51 +19,10 @@ permissions:
|
|||||||
contents: write
|
contents: write
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
deploy-test:
|
|
||||||
name: 🚀 Deploy to test env (fly.io)
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
if: ${{ github.base_ref == 'main' || (github.ref == 'refs/heads/main' && github.event_name == 'push') }}
|
|
||||||
steps:
|
|
||||||
- name: ⬇️ Checkout repo
|
|
||||||
uses: actions/checkout@v3
|
|
||||||
- uses: superfly/flyctl-actions/setup-flyctl@master
|
|
||||||
|
|
||||||
- name: 👀 Read app name
|
|
||||||
uses: SebRollen/toml-action@v1.0.0
|
|
||||||
id: app_name
|
|
||||||
with:
|
|
||||||
file: "fly.toml"
|
|
||||||
field: "app"
|
|
||||||
|
|
||||||
- name: 🚀 Deploy Test
|
|
||||||
run: flyctl deploy --remote-only --wait-timeout=300 --ha=false
|
|
||||||
env:
|
|
||||||
FLY_API_TOKEN: ${{ secrets.FLY_API_TOKEN }}
|
|
||||||
|
|
||||||
manual-approval:
|
|
||||||
name: Manual Approval
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
needs: deploy-test
|
|
||||||
if: success()
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
issues: write
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Await Manual Approval
|
|
||||||
uses: trstringer/manual-approval@v1
|
|
||||||
with:
|
|
||||||
secret: ${{ github.TOKEN }}
|
|
||||||
approvers: DmitryPopov
|
|
||||||
minimum-approvals: 1
|
|
||||||
issue-title: "Manual Approval Required for Release"
|
|
||||||
issue-body: "Please approve or deny the deployment."
|
|
||||||
|
|
||||||
build:
|
build:
|
||||||
name: 🛠 Build
|
name: 🛠 Build
|
||||||
needs: manual-approval
|
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
if: ${{ (github.ref == 'refs/heads/main') && github.event_name == 'push' }}
|
if: ${{ (github.ref == 'refs/heads/main' || github.ref == 'refs/heads/develop') && github.event_name == 'push' }}
|
||||||
permissions:
|
permissions:
|
||||||
checks: write
|
checks: write
|
||||||
contents: write
|
contents: write
|
||||||
@@ -77,7 +37,7 @@ jobs:
|
|||||||
elixir: ["1.17"]
|
elixir: ["1.17"]
|
||||||
node-version: ["18.x"]
|
node-version: ["18.x"]
|
||||||
outputs:
|
outputs:
|
||||||
commit_hash: ${{ steps.generate-changelog.outputs.commit_hash }}
|
commit_hash: ${{ steps.generate-changelog.outputs.commit_hash || steps.set-commit-develop.outputs.commit_hash }}
|
||||||
steps:
|
steps:
|
||||||
- name: Prepare
|
- name: Prepare
|
||||||
run: |
|
run: |
|
||||||
@@ -93,6 +53,7 @@ jobs:
|
|||||||
- name: ⬇️ Checkout repo
|
- name: ⬇️ Checkout repo
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
|
ssh-key: "${{ secrets.COMMIT_KEY }}"
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: 😅 Cache deps
|
- name: 😅 Cache deps
|
||||||
id: cache-deps
|
id: cache-deps
|
||||||
@@ -130,20 +91,26 @@ jobs:
|
|||||||
|
|
||||||
- name: Generate Changelog & Update Tag Version
|
- name: Generate Changelog & Update Tag Version
|
||||||
id: generate-changelog
|
id: generate-changelog
|
||||||
|
if: github.ref == 'refs/heads/main'
|
||||||
run: |
|
run: |
|
||||||
git config --global user.name 'CI'
|
git config --global user.name 'CI'
|
||||||
git config --global user.email 'ci@users.noreply.github.com'
|
git config --global user.email 'ci@users.noreply.github.com'
|
||||||
mix git_ops.release --force-patch --yes
|
mix git_ops.release --force-patch --yes
|
||||||
|
git commit --allow-empty -m 'chore: [skip ci]'
|
||||||
git push --follow-tags
|
git push --follow-tags
|
||||||
echo "commit_hash=$(git rev-parse HEAD)" >> $GITHUB_OUTPUT
|
echo "commit_hash=$(git rev-parse HEAD)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Set commit hash for develop
|
||||||
|
id: set-commit-develop
|
||||||
|
if: github.ref == 'refs/heads/develop'
|
||||||
|
run: |
|
||||||
|
echo "commit_hash=$(git rev-parse HEAD)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
docker:
|
docker:
|
||||||
name: 🛠 Build Docker Images
|
name: 🛠 Build Docker Images
|
||||||
|
if: github.ref == 'refs/heads/develop'
|
||||||
needs: build
|
needs: build
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
outputs:
|
|
||||||
release-tag: ${{ steps.get-latest-tag.outputs.tag }}
|
|
||||||
release-notes: ${{ steps.get-content.outputs.string }}
|
|
||||||
permissions:
|
permissions:
|
||||||
checks: write
|
checks: write
|
||||||
contents: write
|
contents: write
|
||||||
@@ -170,17 +137,6 @@ jobs:
|
|||||||
ref: ${{ needs.build.outputs.commit_hash }}
|
ref: ${{ needs.build.outputs.commit_hash }}
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Prepare Changelog
|
|
||||||
run: |
|
|
||||||
yes | cp -rf CHANGELOG.md priv/changelog/CHANGELOG.md
|
|
||||||
sed -i '1i%{title: "Change Log"}\n\n---\n' priv/changelog/CHANGELOG.md
|
|
||||||
|
|
||||||
- name: Get Release Tag
|
|
||||||
id: get-latest-tag
|
|
||||||
uses: "WyriHaximus/github-action-get-previous-tag@v1"
|
|
||||||
with:
|
|
||||||
fallback: 1.0.0
|
|
||||||
|
|
||||||
- name: Extract metadata (tags, labels) for Docker
|
- name: Extract metadata (tags, labels) for Docker
|
||||||
id: meta
|
id: meta
|
||||||
uses: docker/metadata-action@v5
|
uses: docker/metadata-action@v5
|
||||||
@@ -229,24 +185,6 @@ jobs:
|
|||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
retention-days: 1
|
retention-days: 1
|
||||||
|
|
||||||
- uses: markpatterson27/markdown-to-output@v1
|
|
||||||
id: extract-changelog
|
|
||||||
with:
|
|
||||||
filepath: CHANGELOG.md
|
|
||||||
|
|
||||||
- name: Get content
|
|
||||||
uses: 2428392/gh-truncate-string-action@v1.3.0
|
|
||||||
id: get-content
|
|
||||||
with:
|
|
||||||
stringToTruncate: |
|
|
||||||
📣 Wanderer new release available 🎉
|
|
||||||
|
|
||||||
**Version**: ${{ steps.get-latest-tag.outputs.tag }}
|
|
||||||
|
|
||||||
${{ steps.extract-changelog.outputs.body }}
|
|
||||||
maxLength: 500
|
|
||||||
truncationSymbol: "…"
|
|
||||||
|
|
||||||
merge:
|
merge:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
needs:
|
needs:
|
||||||
@@ -277,9 +215,8 @@ jobs:
|
|||||||
tags: |
|
tags: |
|
||||||
type=ref,event=branch
|
type=ref,event=branch
|
||||||
type=ref,event=pr
|
type=ref,event=pr
|
||||||
type=semver,pattern={{version}}
|
type=raw,value=develop,enable=${{ github.ref == 'refs/heads/develop' }}
|
||||||
type=semver,pattern={{major}}.{{minor}}
|
type=raw,value=develop-{{sha}},enable=${{ github.ref == 'refs/heads/develop' }}
|
||||||
type=semver,pattern={{version}},value=${{ needs.docker.outputs.release-tag }}
|
|
||||||
|
|
||||||
- name: Create manifest list and push
|
- name: Create manifest list and push
|
||||||
working-directory: /tmp/digests
|
working-directory: /tmp/digests
|
||||||
@@ -294,19 +231,25 @@ jobs:
|
|||||||
create-release:
|
create-release:
|
||||||
name: 🏷 Create Release
|
name: 🏷 Create Release
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
needs: [docker, merge]
|
|
||||||
if: ${{ github.ref == 'refs/heads/main' && github.event_name == 'push' }}
|
if: ${{ github.ref == 'refs/heads/main' && github.event_name == 'push' }}
|
||||||
|
needs: build
|
||||||
steps:
|
steps:
|
||||||
- name: ⬇️ Checkout repo
|
- name: ⬇️ Checkout repo
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Get Release Tag
|
||||||
|
id: get-latest-tag
|
||||||
|
uses: "WyriHaximus/github-action-get-previous-tag@v1"
|
||||||
|
with:
|
||||||
|
fallback: 1.0.0
|
||||||
|
|
||||||
- name: 🏷 Create Draft Release
|
- name: 🏷 Create Draft Release
|
||||||
uses: softprops/action-gh-release@v1
|
uses: softprops/action-gh-release@v1
|
||||||
with:
|
with:
|
||||||
tag_name: ${{ needs.docker.outputs.release-tag }}
|
tag_name: ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
name: Release ${{ needs.docker.outputs.release-tag }}
|
name: Release ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
body: |
|
body: |
|
||||||
## Info
|
## Info
|
||||||
Commit ${{ github.sha }} was deployed to `staging`. [See code diff](${{ github.event.compare }}).
|
Commit ${{ github.sha }} was deployed to `staging`. [See code diff](${{ github.event.compare }}).
|
||||||
@@ -316,9 +259,3 @@ jobs:
|
|||||||
## How to Promote?
|
## How to Promote?
|
||||||
In order to promote this to prod, edit the draft and press **"Publish release"**.
|
In order to promote this to prod, edit the draft and press **"Publish release"**.
|
||||||
draft: true
|
draft: true
|
||||||
|
|
||||||
- name: Discord Webhook Action
|
|
||||||
uses: tsickert/discord-webhook@v5.3.0
|
|
||||||
with:
|
|
||||||
webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
|
|
||||||
content: ${{ needs.docker.outputs.release-notes }}
|
|
||||||
|
|||||||
187
.github/workflows/docker-arm.yml
vendored
Normal file
187
.github/workflows/docker-arm.yml
vendored
Normal file
@@ -0,0 +1,187 @@
|
|||||||
|
name: Build Docker ARM Image
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- '**'
|
||||||
|
|
||||||
|
env:
|
||||||
|
MIX_ENV: prod
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
REGISTRY_IMAGE: wandererltd/community-edition-arm
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: ${{ github.workflow }}-${{ github.ref }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
docker:
|
||||||
|
name: 🛠 Build Docker Images
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
outputs:
|
||||||
|
release-tag: ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
|
release-notes: ${{ steps.get-content.outputs.string }}
|
||||||
|
permissions:
|
||||||
|
checks: write
|
||||||
|
contents: write
|
||||||
|
packages: write
|
||||||
|
attestations: write
|
||||||
|
id-token: write
|
||||||
|
pull-requests: write
|
||||||
|
repository-projects: write
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
platform:
|
||||||
|
- linux/arm64
|
||||||
|
steps:
|
||||||
|
- name: Prepare
|
||||||
|
run: |
|
||||||
|
platform=${{ matrix.platform }}
|
||||||
|
echo "PLATFORM_PAIR=${platform//\//-}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: ⬇️ Checkout repo
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Get Release Tag
|
||||||
|
id: get-latest-tag
|
||||||
|
uses: "WyriHaximus/github-action-get-previous-tag@v1"
|
||||||
|
with:
|
||||||
|
fallback: 1.0.0
|
||||||
|
|
||||||
|
- name: ⬇️ Checkout repo
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
ref: ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Prepare Changelog
|
||||||
|
run: |
|
||||||
|
yes | cp -rf CHANGELOG.md priv/changelog/CHANGELOG.md
|
||||||
|
sed -i '1i%{title: "Change Log"}\n\n---\n' priv/changelog/CHANGELOG.md
|
||||||
|
|
||||||
|
- name: Extract metadata (tags, labels) for Docker
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: ${{ env.REGISTRY_IMAGE }}
|
||||||
|
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v3
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Login to DockerHub
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.WANDERER_DOCKER_USER }}
|
||||||
|
password: ${{ secrets.WANDERER_DOCKER_PASSWORD }}
|
||||||
|
|
||||||
|
- name: Build and push
|
||||||
|
id: build
|
||||||
|
uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
push: true
|
||||||
|
context: .
|
||||||
|
file: ./Dockerfile
|
||||||
|
cache-from: type=gha
|
||||||
|
cache-to: type=gha,mode=max
|
||||||
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
|
platforms: ${{ matrix.platform }}
|
||||||
|
outputs: type=image,"name=${{ env.REGISTRY_IMAGE }}",push-by-digest=true,name-canonical=true,push=true
|
||||||
|
build-args: |
|
||||||
|
MIX_ENV=prod
|
||||||
|
BUILD_METADATA=${{ steps.meta.outputs.json }}
|
||||||
|
|
||||||
|
- name: Export digest
|
||||||
|
run: |
|
||||||
|
mkdir -p /tmp/digests
|
||||||
|
digest="${{ steps.build.outputs.digest }}"
|
||||||
|
touch "/tmp/digests/${digest#sha256:}"
|
||||||
|
|
||||||
|
- name: Upload digest
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: digests-${{ env.PLATFORM_PAIR }}
|
||||||
|
path: /tmp/digests/*
|
||||||
|
if-no-files-found: error
|
||||||
|
retention-days: 1
|
||||||
|
|
||||||
|
- uses: markpatterson27/markdown-to-output@v1
|
||||||
|
id: extract-changelog
|
||||||
|
with:
|
||||||
|
filepath: CHANGELOG.md
|
||||||
|
|
||||||
|
- name: Get content
|
||||||
|
uses: 2428392/gh-truncate-string-action@v1.3.0
|
||||||
|
id: get-content
|
||||||
|
with:
|
||||||
|
stringToTruncate: |
|
||||||
|
📣 Wanderer **ARM** release available 🎉
|
||||||
|
|
||||||
|
**Version**: :${{ steps.get-latest-tag.outputs.tag }}
|
||||||
|
|
||||||
|
${{ steps.extract-changelog.outputs.body }}
|
||||||
|
maxLength: 500
|
||||||
|
truncationSymbol: "…"
|
||||||
|
|
||||||
|
merge:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs:
|
||||||
|
- docker
|
||||||
|
steps:
|
||||||
|
- name: Download digests
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
with:
|
||||||
|
path: /tmp/digests
|
||||||
|
pattern: digests-*
|
||||||
|
merge-multiple: true
|
||||||
|
|
||||||
|
- name: Login to Docker Hub
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.WANDERER_DOCKER_USER }}
|
||||||
|
password: ${{ secrets.WANDERER_DOCKER_PASSWORD }}
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Docker meta
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: |
|
||||||
|
${{ env.REGISTRY_IMAGE }}
|
||||||
|
tags: |
|
||||||
|
type=ref,event=branch
|
||||||
|
type=ref,event=pr
|
||||||
|
type=semver,pattern={{version}}
|
||||||
|
type=semver,pattern={{major}}.{{minor}}
|
||||||
|
type=semver,pattern={{version}},value=${{ needs.docker.outputs.release-tag }}
|
||||||
|
|
||||||
|
- name: Create manifest list and push
|
||||||
|
working-directory: /tmp/digests
|
||||||
|
run: |
|
||||||
|
docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \
|
||||||
|
$(printf '${{ env.REGISTRY_IMAGE }}@sha256:%s ' *)
|
||||||
|
|
||||||
|
- name: Inspect image
|
||||||
|
run: |
|
||||||
|
docker buildx imagetools inspect ${{ env.REGISTRY_IMAGE }}:${{ steps.meta.outputs.version }}
|
||||||
|
|
||||||
|
notify:
|
||||||
|
name: 🏷 Notify about release
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
needs: [docker, merge]
|
||||||
|
steps:
|
||||||
|
- name: Discord Webhook Action
|
||||||
|
uses: tsickert/discord-webhook@v5.3.0
|
||||||
|
with:
|
||||||
|
webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
|
||||||
|
content: ${{ needs.docker.outputs.release-notes }}
|
||||||
187
.github/workflows/docker.yml
vendored
Normal file
187
.github/workflows/docker.yml
vendored
Normal file
@@ -0,0 +1,187 @@
|
|||||||
|
name: Build Docker Image
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- '**'
|
||||||
|
|
||||||
|
env:
|
||||||
|
MIX_ENV: prod
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
REGISTRY_IMAGE: wandererltd/community-edition
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: ${{ github.workflow }}-${{ github.ref }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
docker:
|
||||||
|
name: 🛠 Build Docker Images
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
outputs:
|
||||||
|
release-tag: ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
|
release-notes: ${{ steps.get-content.outputs.string }}
|
||||||
|
permissions:
|
||||||
|
checks: write
|
||||||
|
contents: write
|
||||||
|
packages: write
|
||||||
|
attestations: write
|
||||||
|
id-token: write
|
||||||
|
pull-requests: write
|
||||||
|
repository-projects: write
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
platform:
|
||||||
|
- linux/amd64
|
||||||
|
steps:
|
||||||
|
- name: Prepare
|
||||||
|
run: |
|
||||||
|
platform=${{ matrix.platform }}
|
||||||
|
echo "PLATFORM_PAIR=${platform//\//-}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: ⬇️ Checkout repo
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Get Release Tag
|
||||||
|
id: get-latest-tag
|
||||||
|
uses: "WyriHaximus/github-action-get-previous-tag@v1"
|
||||||
|
with:
|
||||||
|
fallback: 1.0.0
|
||||||
|
|
||||||
|
- name: ⬇️ Checkout repo
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
ref: ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
|
- name: Prepare Changelog
|
||||||
|
run: |
|
||||||
|
yes | cp -rf CHANGELOG.md priv/changelog/CHANGELOG.md
|
||||||
|
sed -i '1i%{title: "Change Log"}\n\n---\n' priv/changelog/CHANGELOG.md
|
||||||
|
|
||||||
|
- name: Extract metadata (tags, labels) for Docker
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: ${{ env.REGISTRY_IMAGE }}
|
||||||
|
|
||||||
|
- name: Set up QEMU
|
||||||
|
uses: docker/setup-qemu-action@v3
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Login to DockerHub
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.WANDERER_DOCKER_USER }}
|
||||||
|
password: ${{ secrets.WANDERER_DOCKER_PASSWORD }}
|
||||||
|
|
||||||
|
- name: Build and push
|
||||||
|
id: build
|
||||||
|
uses: docker/build-push-action@v6
|
||||||
|
with:
|
||||||
|
push: true
|
||||||
|
context: .
|
||||||
|
file: ./Dockerfile
|
||||||
|
cache-from: type=gha
|
||||||
|
cache-to: type=gha,mode=max
|
||||||
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
|
platforms: ${{ matrix.platform }}
|
||||||
|
outputs: type=image,"name=${{ env.REGISTRY_IMAGE }}",push-by-digest=true,name-canonical=true,push=true
|
||||||
|
build-args: |
|
||||||
|
MIX_ENV=prod
|
||||||
|
BUILD_METADATA=${{ steps.meta.outputs.json }}
|
||||||
|
|
||||||
|
- name: Export digest
|
||||||
|
run: |
|
||||||
|
mkdir -p /tmp/digests
|
||||||
|
digest="${{ steps.build.outputs.digest }}"
|
||||||
|
touch "/tmp/digests/${digest#sha256:}"
|
||||||
|
|
||||||
|
- name: Upload digest
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: digests-${{ env.PLATFORM_PAIR }}
|
||||||
|
path: /tmp/digests/*
|
||||||
|
if-no-files-found: error
|
||||||
|
retention-days: 1
|
||||||
|
|
||||||
|
- uses: markpatterson27/markdown-to-output@v1
|
||||||
|
id: extract-changelog
|
||||||
|
with:
|
||||||
|
filepath: CHANGELOG.md
|
||||||
|
|
||||||
|
- name: Get content
|
||||||
|
uses: 2428392/gh-truncate-string-action@v1.3.0
|
||||||
|
id: get-content
|
||||||
|
with:
|
||||||
|
stringToTruncate: |
|
||||||
|
📣 Wanderer new release available 🎉
|
||||||
|
|
||||||
|
**Version**: ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
|
|
||||||
|
${{ steps.extract-changelog.outputs.body }}
|
||||||
|
maxLength: 500
|
||||||
|
truncationSymbol: "…"
|
||||||
|
|
||||||
|
merge:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs:
|
||||||
|
- docker
|
||||||
|
steps:
|
||||||
|
- name: Download digests
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
with:
|
||||||
|
path: /tmp/digests
|
||||||
|
pattern: digests-*
|
||||||
|
merge-multiple: true
|
||||||
|
|
||||||
|
- name: Login to Docker Hub
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.WANDERER_DOCKER_USER }}
|
||||||
|
password: ${{ secrets.WANDERER_DOCKER_PASSWORD }}
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Docker meta
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: |
|
||||||
|
${{ env.REGISTRY_IMAGE }}
|
||||||
|
tags: |
|
||||||
|
type=ref,event=branch
|
||||||
|
type=ref,event=pr
|
||||||
|
type=semver,pattern={{version}}
|
||||||
|
type=semver,pattern={{major}}.{{minor}}
|
||||||
|
type=semver,pattern={{version}},value=${{ needs.docker.outputs.release-tag }}
|
||||||
|
|
||||||
|
- name: Create manifest list and push
|
||||||
|
working-directory: /tmp/digests
|
||||||
|
run: |
|
||||||
|
docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \
|
||||||
|
$(printf '${{ env.REGISTRY_IMAGE }}@sha256:%s ' *)
|
||||||
|
|
||||||
|
- name: Inspect image
|
||||||
|
run: |
|
||||||
|
docker buildx imagetools inspect ${{ env.REGISTRY_IMAGE }}:${{ steps.meta.outputs.version }}
|
||||||
|
|
||||||
|
notify:
|
||||||
|
name: 🏷 Notify about release
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
needs: [docker, merge]
|
||||||
|
steps:
|
||||||
|
- name: Discord Webhook Action
|
||||||
|
uses: tsickert/discord-webhook@v5.3.0
|
||||||
|
with:
|
||||||
|
webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
|
||||||
|
content: ${{ needs.docker.outputs.release-notes }}
|
||||||
300
.github/workflows/flaky-test-detection.yml
vendored
Normal file
300
.github/workflows/flaky-test-detection.yml
vendored
Normal file
@@ -0,0 +1,300 @@
|
|||||||
|
name: Flaky Test Detection
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
# Run nightly at 2 AM UTC
|
||||||
|
- cron: '0 2 * * *'
|
||||||
|
workflow_dispatch:
|
||||||
|
inputs:
|
||||||
|
test_file:
|
||||||
|
description: 'Specific test file to check (optional)'
|
||||||
|
required: false
|
||||||
|
type: string
|
||||||
|
iterations:
|
||||||
|
description: 'Number of test iterations'
|
||||||
|
required: false
|
||||||
|
default: '10'
|
||||||
|
type: string
|
||||||
|
|
||||||
|
env:
|
||||||
|
MIX_ENV: test
|
||||||
|
ELIXIR_VERSION: "1.17"
|
||||||
|
OTP_VERSION: "27"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
detect-flaky-tests:
|
||||||
|
name: 🔍 Detect Flaky Tests
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
|
||||||
|
services:
|
||||||
|
postgres:
|
||||||
|
image: postgres:16
|
||||||
|
env:
|
||||||
|
POSTGRES_USER: postgres
|
||||||
|
POSTGRES_PASSWORD: postgres
|
||||||
|
POSTGRES_DB: wanderer_test
|
||||||
|
options: >-
|
||||||
|
--health-cmd pg_isready
|
||||||
|
--health-interval 10s
|
||||||
|
--health-timeout 5s
|
||||||
|
--health-retries 5
|
||||||
|
ports:
|
||||||
|
- 5432:5432
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: ⬇️ Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: 🏗️ Setup Elixir & Erlang
|
||||||
|
uses: erlef/setup-beam@v1
|
||||||
|
with:
|
||||||
|
elixir-version: ${{ env.ELIXIR_VERSION }}
|
||||||
|
otp-version: ${{ env.OTP_VERSION }}
|
||||||
|
|
||||||
|
- name: 📦 Restore dependencies cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
id: deps-cache
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
deps
|
||||||
|
_build
|
||||||
|
key: ${{ runner.os }}-mix-${{ env.ELIXIR_VERSION }}-${{ env.OTP_VERSION }}-${{ hashFiles('**/mix.lock') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-mix-${{ env.ELIXIR_VERSION }}-${{ env.OTP_VERSION }}-
|
||||||
|
|
||||||
|
- name: 📦 Install dependencies
|
||||||
|
if: steps.deps-cache.outputs.cache-hit != 'true'
|
||||||
|
run: |
|
||||||
|
mix deps.get
|
||||||
|
mix deps.compile
|
||||||
|
|
||||||
|
- name: 🏗️ Compile project
|
||||||
|
run: mix compile --warnings-as-errors
|
||||||
|
|
||||||
|
- name: 🏗️ Setup test database
|
||||||
|
run: |
|
||||||
|
mix ecto.create
|
||||||
|
mix ecto.migrate
|
||||||
|
env:
|
||||||
|
DATABASE_URL: postgres://postgres:postgres@localhost:5432/wanderer_test
|
||||||
|
|
||||||
|
- name: 🔍 Run flaky test detection
|
||||||
|
id: flaky-detection
|
||||||
|
run: |
|
||||||
|
# Determine test target
|
||||||
|
TEST_FILE="${{ github.event.inputs.test_file }}"
|
||||||
|
ITERATIONS="${{ github.event.inputs.iterations || '10' }}"
|
||||||
|
|
||||||
|
if [ -n "$TEST_FILE" ]; then
|
||||||
|
echo "Checking specific file: $TEST_FILE"
|
||||||
|
mix test.stability --runs $ITERATIONS --file "$TEST_FILE" --detect --report flaky_report.json
|
||||||
|
else
|
||||||
|
echo "Checking all tests"
|
||||||
|
mix test.stability --runs $ITERATIONS --detect --report flaky_report.json
|
||||||
|
fi
|
||||||
|
env:
|
||||||
|
DATABASE_URL: postgres://postgres:postgres@localhost:5432/wanderer_test
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: 📊 Upload flaky test report
|
||||||
|
if: always()
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: flaky-test-report
|
||||||
|
path: flaky_report.json
|
||||||
|
retention-days: 30
|
||||||
|
|
||||||
|
- name: 💬 Comment on flaky tests
|
||||||
|
if: always()
|
||||||
|
uses: actions/github-script@v7
|
||||||
|
with:
|
||||||
|
script: |
|
||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
// Read the report
|
||||||
|
let report;
|
||||||
|
try {
|
||||||
|
const reportContent = fs.readFileSync('flaky_report.json', 'utf8');
|
||||||
|
report = JSON.parse(reportContent);
|
||||||
|
} catch (error) {
|
||||||
|
console.log('No flaky test report found');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!report.flaky_tests || report.flaky_tests.length === 0) {
|
||||||
|
console.log('No flaky tests detected!');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create issue body
|
||||||
|
const issueBody = `## 🔍 Flaky Tests Detected
|
||||||
|
|
||||||
|
The automated flaky test detection found ${report.flaky_tests.length} potentially flaky test(s).
|
||||||
|
|
||||||
|
### Summary
|
||||||
|
- **Total test runs**: ${report.summary.total_runs}
|
||||||
|
- **Success rate**: ${(report.summary.success_rate * 100).toFixed(1)}%
|
||||||
|
- **Average duration**: ${(report.summary.avg_duration_ms / 1000).toFixed(2)}s
|
||||||
|
|
||||||
|
### Flaky Tests
|
||||||
|
|
||||||
|
| Test | Failure Rate | Details |
|
||||||
|
|------|--------------|---------|
|
||||||
|
${report.flaky_tests.map(test =>
|
||||||
|
`| ${test.test} | ${(test.failure_rate * 100).toFixed(1)}% | Failed ${test.failures}/${report.summary.total_runs} runs |`
|
||||||
|
).join('\n')}
|
||||||
|
|
||||||
|
### Recommended Actions
|
||||||
|
|
||||||
|
1. Review the identified tests for race conditions
|
||||||
|
2. Check for timing dependencies or async issues
|
||||||
|
3. Ensure proper test isolation and cleanup
|
||||||
|
4. Consider adding explicit waits or synchronization
|
||||||
|
5. Use \`async: false\` if tests share resources
|
||||||
|
|
||||||
|
---
|
||||||
|
*This issue was automatically created by the flaky test detection workflow.*
|
||||||
|
*Run time: ${new Date().toISOString()}*
|
||||||
|
`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Check if there's already an open issue
|
||||||
|
const issues = await github.rest.issues.listForRepo({
|
||||||
|
owner: context.repo.owner,
|
||||||
|
repo: context.repo.repo,
|
||||||
|
labels: 'flaky-test',
|
||||||
|
state: 'open'
|
||||||
|
});
|
||||||
|
|
||||||
|
if (issues.data.length > 0) {
|
||||||
|
// Update existing issue
|
||||||
|
const issue = issues.data[0];
|
||||||
|
try {
|
||||||
|
await github.rest.issues.createComment({
|
||||||
|
owner: context.repo.owner,
|
||||||
|
repo: context.repo.repo,
|
||||||
|
issue_number: issue.number,
|
||||||
|
body: issueBody
|
||||||
|
});
|
||||||
|
console.log(`Updated existing issue #${issue.number}`);
|
||||||
|
} catch (commentError) {
|
||||||
|
console.error('Failed to create comment:', commentError.message);
|
||||||
|
throw commentError;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Create new issue
|
||||||
|
try {
|
||||||
|
const newIssue = await github.rest.issues.create({
|
||||||
|
owner: context.repo.owner,
|
||||||
|
repo: context.repo.repo,
|
||||||
|
title: '🔍 Flaky Tests Detected',
|
||||||
|
body: issueBody,
|
||||||
|
labels: ['flaky-test', 'test-quality', 'automated']
|
||||||
|
});
|
||||||
|
console.log(`Created new issue #${newIssue.data.number}`);
|
||||||
|
} catch (createError) {
|
||||||
|
console.error('Failed to create issue:', createError.message);
|
||||||
|
throw createError;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (listError) {
|
||||||
|
console.error('Failed to list issues:', listError.message);
|
||||||
|
console.error('API error details:', listError.response?.data || 'No response data');
|
||||||
|
throw listError;
|
||||||
|
}
|
||||||
|
|
||||||
|
- name: 📈 Update metrics
|
||||||
|
if: always()
|
||||||
|
run: |
|
||||||
|
# Parse and store metrics for tracking
|
||||||
|
if [ -f flaky_report.json ]; then
|
||||||
|
FLAKY_COUNT=$(jq '.flaky_tests | length' flaky_report.json)
|
||||||
|
SUCCESS_RATE=$(jq '.summary.success_rate' flaky_report.json)
|
||||||
|
|
||||||
|
echo "FLAKY_TEST_COUNT=$FLAKY_COUNT" >> $GITHUB_ENV
|
||||||
|
echo "TEST_SUCCESS_RATE=$SUCCESS_RATE" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
# Log metrics (could be sent to monitoring service)
|
||||||
|
echo "::notice title=Flaky Test Metrics::Found $FLAKY_COUNT flaky tests with ${SUCCESS_RATE}% success rate"
|
||||||
|
fi
|
||||||
|
|
||||||
|
analyze-test-history:
|
||||||
|
name: 📊 Analyze Test History
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
needs: detect-flaky-tests
|
||||||
|
if: always()
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: ⬇️ Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: 📥 Download previous reports
|
||||||
|
uses: dawidd6/action-download-artifact@v3
|
||||||
|
with:
|
||||||
|
workflow: flaky-test-detection.yml
|
||||||
|
workflow_conclusion: completed
|
||||||
|
name: flaky-test-report
|
||||||
|
path: historical-reports
|
||||||
|
if_no_artifact_found: warn
|
||||||
|
|
||||||
|
- name: 📊 Generate trend analysis
|
||||||
|
run: |
|
||||||
|
# Analyze historical trends
|
||||||
|
python3 <<'EOF'
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
from datetime import datetime
|
||||||
|
import glob
|
||||||
|
|
||||||
|
reports = []
|
||||||
|
for report_file in glob.glob('historical-reports/*/flaky_report.json'):
|
||||||
|
try:
|
||||||
|
with open(report_file, 'r') as f:
|
||||||
|
data = json.load(f)
|
||||||
|
reports.append(data)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
if not reports:
|
||||||
|
print("No historical data found")
|
||||||
|
exit(0)
|
||||||
|
|
||||||
|
# Sort by timestamp
|
||||||
|
reports.sort(key=lambda x: x.get('timestamp', ''), reverse=True)
|
||||||
|
|
||||||
|
# Analyze trends
|
||||||
|
print("## Test Stability Trend Analysis")
|
||||||
|
print(f"\nAnalyzed {len(reports)} historical reports")
|
||||||
|
print("\n### Flaky Test Counts Over Time")
|
||||||
|
|
||||||
|
for report in reports[:10]: # Last 10 reports
|
||||||
|
timestamp = report.get('timestamp', 'Unknown')
|
||||||
|
flaky_count = len(report.get('flaky_tests', []))
|
||||||
|
success_rate = report.get('summary', {}).get('success_rate', 0) * 100
|
||||||
|
print(f"- {timestamp[:10]}: {flaky_count} flaky tests ({success_rate:.1f}% success rate)")
|
||||||
|
|
||||||
|
# Identify persistently flaky tests
|
||||||
|
all_flaky = {}
|
||||||
|
for report in reports:
|
||||||
|
for test in report.get('flaky_tests', []):
|
||||||
|
test_name = test.get('test', '')
|
||||||
|
if test_name not in all_flaky:
|
||||||
|
all_flaky[test_name] = 0
|
||||||
|
all_flaky[test_name] += 1
|
||||||
|
|
||||||
|
if all_flaky:
|
||||||
|
print("\n### Persistently Flaky Tests")
|
||||||
|
sorted_flaky = sorted(all_flaky.items(), key=lambda x: x[1], reverse=True)
|
||||||
|
for test_name, count in sorted_flaky[:5]:
|
||||||
|
percentage = (count / len(reports)) * 100
|
||||||
|
print(f"- {test_name}: Flaky in {count}/{len(reports)} runs ({percentage:.1f}%)")
|
||||||
|
EOF
|
||||||
|
|
||||||
|
- name: 💾 Save analysis
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: test-stability-analysis
|
||||||
|
path: |
|
||||||
|
flaky_report.json
|
||||||
|
historical-reports/
|
||||||
|
retention-days: 90
|
||||||
333
.github/workflows/test.yml
vendored
Normal file
333
.github/workflows/test.yml
vendored
Normal file
@@ -0,0 +1,333 @@
|
|||||||
|
name: 🧪 Test Suite
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches: [main, develop]
|
||||||
|
push:
|
||||||
|
branches: [main, develop]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
pull-requests: write
|
||||||
|
issues: write
|
||||||
|
|
||||||
|
env:
|
||||||
|
MIX_ENV: test
|
||||||
|
ELIXIR_VERSION: '1.16'
|
||||||
|
OTP_VERSION: '26'
|
||||||
|
NODE_VERSION: '18'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
name: Test Suite
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
services:
|
||||||
|
postgres:
|
||||||
|
image: postgres:15
|
||||||
|
env:
|
||||||
|
POSTGRES_PASSWORD: postgres
|
||||||
|
POSTGRES_DB: wanderer_test
|
||||||
|
options: >-
|
||||||
|
--health-cmd pg_isready
|
||||||
|
--health-interval 10s
|
||||||
|
--health-timeout 5s
|
||||||
|
--health-retries 5
|
||||||
|
ports:
|
||||||
|
- 5432:5432
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup Elixir/OTP
|
||||||
|
uses: erlef/setup-beam@v1
|
||||||
|
with:
|
||||||
|
elixir-version: ${{ env.ELIXIR_VERSION }}
|
||||||
|
otp-version: ${{ env.OTP_VERSION }}
|
||||||
|
|
||||||
|
- name: Cache Elixir dependencies
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
deps
|
||||||
|
_build
|
||||||
|
key: ${{ runner.os }}-mix-${{ hashFiles('**/mix.lock') }}
|
||||||
|
restore-keys: ${{ runner.os }}-mix-
|
||||||
|
|
||||||
|
- name: Install Elixir dependencies
|
||||||
|
run: |
|
||||||
|
mix deps.get
|
||||||
|
mix deps.compile
|
||||||
|
|
||||||
|
- name: Check code formatting
|
||||||
|
id: format
|
||||||
|
run: |
|
||||||
|
if mix format --check-formatted; then
|
||||||
|
echo "status=✅ Passed" >> $GITHUB_OUTPUT
|
||||||
|
echo "count=0" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Failed" >> $GITHUB_OUTPUT
|
||||||
|
echo "count=1" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Compile code and capture warnings
|
||||||
|
id: compile
|
||||||
|
run: |
|
||||||
|
# Capture compilation output
|
||||||
|
output=$(mix compile 2>&1 || true)
|
||||||
|
echo "$output" > compile_output.txt
|
||||||
|
|
||||||
|
# Count warnings
|
||||||
|
warning_count=$(echo "$output" | grep -c "warning:" || echo "0")
|
||||||
|
|
||||||
|
# Check if compilation succeeded
|
||||||
|
if mix compile > /dev/null 2>&1; then
|
||||||
|
echo "status=✅ Success" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Failed" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "warnings=$warning_count" >> $GITHUB_OUTPUT
|
||||||
|
echo "output<<EOF" >> $GITHUB_OUTPUT
|
||||||
|
echo "$output" >> $GITHUB_OUTPUT
|
||||||
|
echo "EOF" >> $GITHUB_OUTPUT
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Setup database
|
||||||
|
run: |
|
||||||
|
mix ecto.create
|
||||||
|
mix ecto.migrate
|
||||||
|
|
||||||
|
- name: Run tests with coverage
|
||||||
|
id: tests
|
||||||
|
run: |
|
||||||
|
# Run tests with coverage
|
||||||
|
output=$(mix test --cover 2>&1 || true)
|
||||||
|
echo "$output" > test_output.txt
|
||||||
|
|
||||||
|
# Parse test results
|
||||||
|
if echo "$output" | grep -q "0 failures"; then
|
||||||
|
echo "status=✅ All Passed" >> $GITHUB_OUTPUT
|
||||||
|
test_status="success"
|
||||||
|
else
|
||||||
|
echo "status=❌ Some Failed" >> $GITHUB_OUTPUT
|
||||||
|
test_status="failed"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Extract test counts
|
||||||
|
test_line=$(echo "$output" | grep -E "[0-9]+ tests?, [0-9]+ failures?" | head -1 || echo "0 tests, 0 failures")
|
||||||
|
total_tests=$(echo "$test_line" | grep -o '[0-9]\+ tests\?' | grep -o '[0-9]\+' | head -1 || echo "0")
|
||||||
|
failures=$(echo "$test_line" | grep -o '[0-9]\+ failures\?' | grep -o '[0-9]\+' | head -1 || echo "0")
|
||||||
|
|
||||||
|
echo "total=$total_tests" >> $GITHUB_OUTPUT
|
||||||
|
echo "failures=$failures" >> $GITHUB_OUTPUT
|
||||||
|
echo "passed=$((total_tests - failures))" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Calculate success rate
|
||||||
|
if [ "$total_tests" -gt 0 ]; then
|
||||||
|
success_rate=$(echo "scale=1; ($total_tests - $failures) * 100 / $total_tests" | bc)
|
||||||
|
else
|
||||||
|
success_rate="0"
|
||||||
|
fi
|
||||||
|
echo "success_rate=$success_rate" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
exit_code=$?
|
||||||
|
echo "exit_code=$exit_code" >> $GITHUB_OUTPUT
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Generate coverage report
|
||||||
|
id: coverage
|
||||||
|
run: |
|
||||||
|
# Generate coverage report with GitHub format
|
||||||
|
output=$(mix coveralls.github 2>&1 || true)
|
||||||
|
echo "$output" > coverage_output.txt
|
||||||
|
|
||||||
|
# Extract coverage percentage
|
||||||
|
coverage=$(echo "$output" | grep -o '[0-9]\+\.[0-9]\+%' | head -1 | sed 's/%//' || echo "0")
|
||||||
|
if [ -z "$coverage" ]; then
|
||||||
|
coverage="0"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "percentage=$coverage" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Determine status
|
||||||
|
if (( $(echo "$coverage >= 80" | bc -l) )); then
|
||||||
|
echo "status=✅ Excellent" >> $GITHUB_OUTPUT
|
||||||
|
elif (( $(echo "$coverage >= 60" | bc -l) )); then
|
||||||
|
echo "status=⚠️ Good" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Needs Improvement" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Run Credo analysis
|
||||||
|
id: credo
|
||||||
|
run: |
|
||||||
|
# Run Credo and capture output
|
||||||
|
output=$(mix credo --strict --format=json 2>&1 || true)
|
||||||
|
echo "$output" > credo_output.txt
|
||||||
|
|
||||||
|
# Try to parse JSON output
|
||||||
|
if echo "$output" | jq . > /dev/null 2>&1; then
|
||||||
|
issues=$(echo "$output" | jq '.issues | length' 2>/dev/null || echo "0")
|
||||||
|
high_issues=$(echo "$output" | jq '.issues | map(select(.priority == "high")) | length' 2>/dev/null || echo "0")
|
||||||
|
normal_issues=$(echo "$output" | jq '.issues | map(select(.priority == "normal")) | length' 2>/dev/null || echo "0")
|
||||||
|
low_issues=$(echo "$output" | jq '.issues | map(select(.priority == "low")) | length' 2>/dev/null || echo "0")
|
||||||
|
else
|
||||||
|
# Fallback: try to count issues from regular output
|
||||||
|
regular_output=$(mix credo --strict 2>&1 || true)
|
||||||
|
issues=$(echo "$regular_output" | grep -c "┃" || echo "0")
|
||||||
|
high_issues="0"
|
||||||
|
normal_issues="0"
|
||||||
|
low_issues="0"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "total_issues=$issues" >> $GITHUB_OUTPUT
|
||||||
|
echo "high_issues=$high_issues" >> $GITHUB_OUTPUT
|
||||||
|
echo "normal_issues=$normal_issues" >> $GITHUB_OUTPUT
|
||||||
|
echo "low_issues=$low_issues" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Determine status
|
||||||
|
if [ "$issues" -eq 0 ]; then
|
||||||
|
echo "status=✅ Clean" >> $GITHUB_OUTPUT
|
||||||
|
elif [ "$issues" -lt 10 ]; then
|
||||||
|
echo "status=⚠️ Minor Issues" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Needs Attention" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Run Dialyzer analysis
|
||||||
|
id: dialyzer
|
||||||
|
run: |
|
||||||
|
# Ensure PLT is built
|
||||||
|
mix dialyzer --plt
|
||||||
|
|
||||||
|
# Run Dialyzer and capture output
|
||||||
|
output=$(mix dialyzer --format=github 2>&1 || true)
|
||||||
|
echo "$output" > dialyzer_output.txt
|
||||||
|
|
||||||
|
# Count warnings and errors
|
||||||
|
warnings=$(echo "$output" | grep -c "warning:" || echo "0")
|
||||||
|
errors=$(echo "$output" | grep -c "error:" || echo "0")
|
||||||
|
|
||||||
|
echo "warnings=$warnings" >> $GITHUB_OUTPUT
|
||||||
|
echo "errors=$errors" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Determine status
|
||||||
|
if [ "$errors" -eq 0 ] && [ "$warnings" -eq 0 ]; then
|
||||||
|
echo "status=✅ Clean" >> $GITHUB_OUTPUT
|
||||||
|
elif [ "$errors" -eq 0 ]; then
|
||||||
|
echo "status=⚠️ Warnings Only" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Has Errors" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Create test results summary
|
||||||
|
id: summary
|
||||||
|
run: |
|
||||||
|
# Calculate overall score
|
||||||
|
format_score=${{ steps.format.outputs.count == '0' && '100' || '0' }}
|
||||||
|
compile_score=${{ steps.compile.outputs.warnings == '0' && '100' || '80' }}
|
||||||
|
test_score=${{ steps.tests.outputs.success_rate }}
|
||||||
|
coverage_score=${{ steps.coverage.outputs.percentage }}
|
||||||
|
credo_score=$(echo "scale=0; (100 - ${{ steps.credo.outputs.total_issues }} * 2)" | bc | sed 's/^-.*$/0/')
|
||||||
|
dialyzer_score=$(echo "scale=0; (100 - ${{ steps.dialyzer.outputs.warnings }} * 2 - ${{ steps.dialyzer.outputs.errors }} * 10)" | bc | sed 's/^-.*$/0/')
|
||||||
|
|
||||||
|
overall_score=$(echo "scale=1; ($format_score + $compile_score + $test_score + $coverage_score + $credo_score + $dialyzer_score) / 6" | bc)
|
||||||
|
|
||||||
|
echo "overall_score=$overall_score" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Determine overall status
|
||||||
|
if (( $(echo "$overall_score >= 90" | bc -l) )); then
|
||||||
|
echo "overall_status=🌟 Excellent" >> $GITHUB_OUTPUT
|
||||||
|
elif (( $(echo "$overall_score >= 80" | bc -l) )); then
|
||||||
|
echo "overall_status=✅ Good" >> $GITHUB_OUTPUT
|
||||||
|
elif (( $(echo "$overall_score >= 70" | bc -l) )); then
|
||||||
|
echo "overall_status=⚠️ Needs Improvement" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "overall_status=❌ Poor" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Find existing PR comment
|
||||||
|
if: github.event_name == 'pull_request'
|
||||||
|
id: find_comment
|
||||||
|
uses: peter-evans/find-comment@v3
|
||||||
|
with:
|
||||||
|
issue-number: ${{ github.event.pull_request.number }}
|
||||||
|
comment-author: 'github-actions[bot]'
|
||||||
|
body-includes: '## 🧪 Test Results Summary'
|
||||||
|
|
||||||
|
- name: Create or update PR comment
|
||||||
|
if: github.event_name == 'pull_request'
|
||||||
|
uses: peter-evans/create-or-update-comment@v4
|
||||||
|
with:
|
||||||
|
comment-id: ${{ steps.find_comment.outputs.comment-id }}
|
||||||
|
issue-number: ${{ github.event.pull_request.number }}
|
||||||
|
edit-mode: replace
|
||||||
|
body: |
|
||||||
|
## 🧪 Test Results Summary
|
||||||
|
|
||||||
|
**Overall Quality Score: ${{ steps.summary.outputs.overall_score }}%** ${{ steps.summary.outputs.overall_status }}
|
||||||
|
|
||||||
|
### 📊 Metrics Dashboard
|
||||||
|
|
||||||
|
| Category | Status | Count | Details |
|
||||||
|
|----------|---------|-------|---------|
|
||||||
|
| 📝 **Code Formatting** | ${{ steps.format.outputs.status }} | ${{ steps.format.outputs.count }} issues | `mix format --check-formatted` |
|
||||||
|
| 🔨 **Compilation** | ${{ steps.compile.outputs.status }} | ${{ steps.compile.outputs.warnings }} warnings | `mix compile` |
|
||||||
|
| 🧪 **Tests** | ${{ steps.tests.outputs.status }} | ${{ steps.tests.outputs.failures }}/${{ steps.tests.outputs.total }} failed | Success rate: ${{ steps.tests.outputs.success_rate }}% |
|
||||||
|
| 📊 **Coverage** | ${{ steps.coverage.outputs.status }} | ${{ steps.coverage.outputs.percentage }}% | `mix coveralls` |
|
||||||
|
| 🎯 **Credo** | ${{ steps.credo.outputs.status }} | ${{ steps.credo.outputs.total_issues }} issues | High: ${{ steps.credo.outputs.high_issues }}, Normal: ${{ steps.credo.outputs.normal_issues }}, Low: ${{ steps.credo.outputs.low_issues }} |
|
||||||
|
| 🔍 **Dialyzer** | ${{ steps.dialyzer.outputs.status }} | ${{ steps.dialyzer.outputs.errors }} errors, ${{ steps.dialyzer.outputs.warnings }} warnings | `mix dialyzer` |
|
||||||
|
|
||||||
|
### 🎯 Quality Gates
|
||||||
|
|
||||||
|
Based on the project's quality thresholds:
|
||||||
|
- **Compilation Warnings**: ${{ steps.compile.outputs.warnings }}/148 (limit: 148)
|
||||||
|
- **Credo Issues**: ${{ steps.credo.outputs.total_issues }}/87 (limit: 87)
|
||||||
|
- **Dialyzer Warnings**: ${{ steps.dialyzer.outputs.warnings }}/161 (limit: 161)
|
||||||
|
- **Test Coverage**: ${{ steps.coverage.outputs.percentage }}%/50% (minimum: 50%)
|
||||||
|
- **Test Failures**: ${{ steps.tests.outputs.failures }}/0 (limit: 0)
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>📈 Progress Toward Goals</summary>
|
||||||
|
|
||||||
|
Target goals for the project:
|
||||||
|
- ✨ **Zero compilation warnings** (currently: ${{ steps.compile.outputs.warnings }})
|
||||||
|
- ✨ **≤10 Credo issues** (currently: ${{ steps.credo.outputs.total_issues }})
|
||||||
|
- ✨ **Zero Dialyzer warnings** (currently: ${{ steps.dialyzer.outputs.warnings }})
|
||||||
|
- ✨ **≥85% test coverage** (currently: ${{ steps.coverage.outputs.percentage }}%)
|
||||||
|
- ✅ **Zero test failures** (currently: ${{ steps.tests.outputs.failures }})
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>🔧 Quick Actions</summary>
|
||||||
|
|
||||||
|
To improve code quality:
|
||||||
|
```bash
|
||||||
|
# Fix formatting issues
|
||||||
|
mix format
|
||||||
|
|
||||||
|
# View detailed Credo analysis
|
||||||
|
mix credo --strict
|
||||||
|
|
||||||
|
# Check Dialyzer warnings
|
||||||
|
mix dialyzer
|
||||||
|
|
||||||
|
# Generate detailed coverage report
|
||||||
|
mix coveralls.html
|
||||||
|
```
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
🤖 *Auto-generated by GitHub Actions* • Updated: ${{ github.event.head_commit.timestamp }}
|
||||||
|
|
||||||
|
> **Note**: This comment will be updated automatically when new commits are pushed to this PR.
|
||||||
7
.gitignore
vendored
7
.gitignore
vendored
@@ -4,7 +4,8 @@
|
|||||||
*.iml
|
*.iml
|
||||||
|
|
||||||
*.key
|
*.key
|
||||||
|
.repomixignore
|
||||||
|
repomix*
|
||||||
/.idea/
|
/.idea/
|
||||||
/node_modules/
|
/node_modules/
|
||||||
/assets/node_modules/
|
/assets/node_modules/
|
||||||
@@ -17,6 +18,9 @@
|
|||||||
/priv/static/*.js
|
/priv/static/*.js
|
||||||
/priv/static/*.css
|
/priv/static/*.css
|
||||||
|
|
||||||
|
# Dialyzer PLT files
|
||||||
|
/priv/plts/
|
||||||
|
|
||||||
.DS_Store
|
.DS_Store
|
||||||
**/.DS_Store
|
**/.DS_Store
|
||||||
|
|
||||||
@@ -26,6 +30,7 @@
|
|||||||
|
|
||||||
.env
|
.env
|
||||||
*.local.env
|
*.local.env
|
||||||
|
test/manual/.auto*
|
||||||
|
|
||||||
.direnv/
|
.direnv/
|
||||||
.cache/
|
.cache/
|
||||||
|
|||||||
2988
CHANGELOG.md
2988
CHANGELOG.md
File diff suppressed because it is too large
Load Diff
@@ -21,21 +21,17 @@ RUN mkdir config
|
|||||||
# to ensure any relevant config change will trigger the dependencies
|
# to ensure any relevant config change will trigger the dependencies
|
||||||
# to be re-compiled.
|
# to be re-compiled.
|
||||||
COPY config/config.exs config/${MIX_ENV}.exs config/
|
COPY config/config.exs config/${MIX_ENV}.exs config/
|
||||||
|
|
||||||
COPY priv priv
|
COPY priv priv
|
||||||
|
|
||||||
COPY lib lib
|
COPY lib lib
|
||||||
|
|
||||||
COPY assets assets
|
COPY assets assets
|
||||||
|
|
||||||
RUN mix compile
|
|
||||||
|
|
||||||
RUN mix assets.deploy
|
RUN mix assets.deploy
|
||||||
|
RUN mix compile
|
||||||
|
|
||||||
# Changes to config/runtime.exs don't require recompiling the code
|
# Changes to config/runtime.exs don't require recompiling the code
|
||||||
COPY config/runtime.exs config/
|
COPY config/runtime.exs config/
|
||||||
|
|
||||||
COPY rel rel
|
COPY rel rel
|
||||||
|
|
||||||
RUN mix release
|
RUN mix release
|
||||||
|
|
||||||
# start a new build stage so that the final image will only contain
|
# start a new build stage so that the final image will only contain
|
||||||
|
|||||||
7
Makefile
7
Makefile
@@ -1,4 +1,4 @@
|
|||||||
.PHONY: deploy install cleanup start yarn migrate format test coverage versions
|
.PHONY: deploy install cleanup start yarn migrate format test coverage versions standalone-tests
|
||||||
|
|
||||||
ROOT_DIR:=$(shell dirname $(realpath $(firstword $(MAKEFILE_LIST))))
|
ROOT_DIR:=$(shell dirname $(realpath $(firstword $(MAKEFILE_LIST))))
|
||||||
SHELL := /bin/bash
|
SHELL := /bin/bash
|
||||||
@@ -35,6 +35,11 @@ test t:
|
|||||||
coverage cover co:
|
coverage cover co:
|
||||||
mix test --cover
|
mix test --cover
|
||||||
|
|
||||||
|
unit-tests ut:
|
||||||
|
@echo "Running unit tests..."
|
||||||
|
@find test/unit -name "*.exs" -exec elixir {} \;
|
||||||
|
@echo "All unit tests completed."
|
||||||
|
|
||||||
versions v:
|
versions v:
|
||||||
@echo "Tool Versions"
|
@echo "Tool Versions"
|
||||||
@cat .tool-versions
|
@cat .tool-versions
|
||||||
|
|||||||
@@ -17,5 +17,6 @@ module.exports = {
|
|||||||
'react-refresh/only-export-components': ['warn', { allowConstantExport: true }],
|
'react-refresh/only-export-components': ['warn', { allowConstantExport: true }],
|
||||||
'react/react-in-jsx-scope': 'off',
|
'react/react-in-jsx-scope': 'off',
|
||||||
'@typescript-eslint/ban-ts-comment': 'off',
|
'@typescript-eslint/ban-ts-comment': 'off',
|
||||||
|
"linebreak-style": "off",
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -7,5 +7,5 @@
|
|||||||
"semi": true,
|
"semi": true,
|
||||||
"tabWidth": 2,
|
"tabWidth": 2,
|
||||||
"useTabs": false,
|
"useTabs": false,
|
||||||
"endOfLine": "lf"
|
"endOfLine": "auto"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -112,19 +112,19 @@ body > div:first-of-type {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.wd-characters-icons {
|
.wd-characters-icons {
|
||||||
display: flex;
|
/*display: flex;*/
|
||||||
transition:
|
/*transition:*/
|
||||||
border-color 250ms,
|
/* border-color 250ms,*/
|
||||||
opacity 250ms;
|
/* opacity 250ms;*/
|
||||||
width: 35px;
|
/*width: 35px;*/
|
||||||
height: 35px;
|
/*height: 35px;*/
|
||||||
border-radius: 50%;
|
/*border-radius: 50%;*/
|
||||||
border-width: 2px;
|
/*border-width: 2px;*/
|
||||||
border-style: solid;
|
/*border-style: solid;*/
|
||||||
border-color: #5a5a5a;
|
/*border-color: #5a5a5a;*/
|
||||||
background-color: rgba(0, 0, 0, 0);
|
/*background-color: rgba(0, 0, 0, 0);*/
|
||||||
cursor: pointer;
|
/*cursor: pointer;*/
|
||||||
opacity: 0.6;
|
/*opacity: 0.6;*/
|
||||||
}
|
}
|
||||||
|
|
||||||
.wd-bg-default {
|
.wd-bg-default {
|
||||||
|
|||||||
14
assets/jest.config.js
Normal file
14
assets/jest.config.js
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
module.exports = {
|
||||||
|
preset: 'ts-jest',
|
||||||
|
testEnvironment: 'jsdom',
|
||||||
|
roots: ['<rootDir>'],
|
||||||
|
moduleDirectories: ['node_modules', 'js'],
|
||||||
|
moduleNameMapper: {
|
||||||
|
'^@/(.*)$': '<rootDir>/js/$1',
|
||||||
|
'\.scss$': 'identity-obj-proxy', // Mock SCSS files
|
||||||
|
},
|
||||||
|
transform: {
|
||||||
|
'^.+\.(ts|tsx)$': 'ts-jest',
|
||||||
|
'^.+\.(js|jsx)$': 'babel-jest', // Add babel-jest for JS/JSX files if needed
|
||||||
|
},
|
||||||
|
};
|
||||||
@@ -1,14 +1,14 @@
|
|||||||
import { ErrorBoundary } from 'react-error-boundary';
|
|
||||||
import { PrimeReactProvider } from 'primereact/api';
|
import { PrimeReactProvider } from 'primereact/api';
|
||||||
|
import { ErrorBoundary } from 'react-error-boundary';
|
||||||
|
|
||||||
import { ReactFlowProvider } from 'reactflow';
|
|
||||||
import { MapHandlers } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { MapHandlers } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { ErrorInfo, useCallback, useEffect, useRef } from 'react';
|
import { ErrorInfo, useCallback, useEffect, useRef } from 'react';
|
||||||
|
import { ReactFlowProvider } from 'reactflow';
|
||||||
import { useMapperHandlers } from './useMapperHandlers';
|
import { useMapperHandlers } from './useMapperHandlers';
|
||||||
|
|
||||||
import './common-styles/main.scss';
|
|
||||||
import { MapRootProvider } from '@/hooks/Mapper/mapRootProvider';
|
|
||||||
import { MapRootContent } from '@/hooks/Mapper/components/mapRootContent/MapRootContent.tsx';
|
import { MapRootContent } from '@/hooks/Mapper/components/mapRootContent/MapRootContent.tsx';
|
||||||
|
import { MapRootProvider } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import './common-styles/main.scss';
|
||||||
|
|
||||||
const ErrorFallback = () => {
|
const ErrorFallback = () => {
|
||||||
return <div className="!z-100 absolute w-screen h-screen bg-transparent"></div>;
|
return <div className="!z-100 absolute w-screen h-screen bg-transparent"></div>;
|
||||||
@@ -20,7 +20,7 @@ export default function MapRoot({ hooks }) {
|
|||||||
|
|
||||||
const mapperHandlerRefs = useRef([providerRef]);
|
const mapperHandlerRefs = useRef([providerRef]);
|
||||||
|
|
||||||
const { handleCommand, handleMapEvent, handleMapEvents } = useMapperHandlers(mapperHandlerRefs.current, hooksRef);
|
const { handleCommand, handleMapEvent } = useMapperHandlers(mapperHandlerRefs.current, hooksRef);
|
||||||
|
|
||||||
const logError = useCallback((error: Error, info: ErrorInfo) => {
|
const logError = useCallback((error: Error, info: ErrorInfo) => {
|
||||||
if (!hooksRef.current) {
|
if (!hooksRef.current) {
|
||||||
@@ -35,7 +35,6 @@ export default function MapRoot({ hooks }) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
hooksRef.current.handleEvent('map_event', handleMapEvent);
|
hooksRef.current.handleEvent('map_event', handleMapEvent);
|
||||||
hooksRef.current.handleEvent('map_events', handleMapEvents);
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -67,16 +67,22 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.p-sortable-column {
|
.p-datatable-thead {
|
||||||
font-size: 12px;
|
th, th.p-sortable-column {
|
||||||
font-weight: bold;
|
font-size: 12px;
|
||||||
padding: 3px 4px;
|
font-weight: bold;
|
||||||
|
padding: 3px 4px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.p-selectable-row td {
|
.p-selectable-row td {
|
||||||
padding: 4px 4px;
|
padding: 4px 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.p-datatable.p-datatable-sm .p-datatable-tbody > tr > td {
|
||||||
|
padding: 3px 4px;
|
||||||
|
}
|
||||||
|
|
||||||
.p-sortable-column > .p-column-header-content > span:last-child {
|
.p-sortable-column > .p-column-header-content > span:last-child {
|
||||||
transform: scale(0.7);
|
transform: scale(0.7);
|
||||||
|
|
||||||
@@ -93,6 +99,11 @@
|
|||||||
.p-dropdown-item {
|
.p-dropdown-item {
|
||||||
padding: 0.25rem 0.5rem;
|
padding: 0.25rem 0.5rem;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
.p-dropdown-item-label {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.p-dropdown-item-group {
|
.p-dropdown-item-group {
|
||||||
@@ -137,3 +148,139 @@
|
|||||||
background: #966d3d;
|
background: #966d3d;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.p-datatable-wrapper {
|
||||||
|
height: 100%;
|
||||||
|
& {
|
||||||
|
scrollbar-width: thin;
|
||||||
|
scrollbar-color: rgba(255, 255, 255, 0.5) transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
&::-webkit-scrollbar {
|
||||||
|
width: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&::-webkit-scrollbar-track {
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
&::-webkit-scrollbar-thumb {
|
||||||
|
background-color: rgba(255, 255, 255, 0.5);
|
||||||
|
border-radius: 5px;
|
||||||
|
border: 2px solid transparent;
|
||||||
|
background-clip: content-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
&::-webkit-scrollbar-thumb:hover {
|
||||||
|
background-color: rgba(255, 255, 255, 0.7);
|
||||||
|
}
|
||||||
|
|
||||||
|
&::-webkit-scrollbar-button {
|
||||||
|
display: none;
|
||||||
|
height: 0;
|
||||||
|
width: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-datatable .p-datatable-tbody > tr.p-highlight {
|
||||||
|
background: initial;
|
||||||
|
}
|
||||||
|
|
||||||
|
.suppress-menu-behaviour {
|
||||||
|
pointer-events: none;
|
||||||
|
|
||||||
|
.p-menuitem-content {
|
||||||
|
pointer-events: initial;
|
||||||
|
background-color: initial !important;
|
||||||
|
}
|
||||||
|
.p-menuitem-content:hover {
|
||||||
|
background-color: initial !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.p-autocomplete .p-autocomplete-multiple-container:not(.p-disabled).p-focus {
|
||||||
|
box-shadow: 0 0 0 1px #335c7e;
|
||||||
|
border-color: #335c7e;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-inputtext:enabled:focus {
|
||||||
|
box-shadow: 0 0 0 1px #335c7e;
|
||||||
|
border-color: #335c7e;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-inputtext:enabled:hover {
|
||||||
|
border-color: #335c7e;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// --------------- TOAST
|
||||||
|
.p-toast .p-toast-message {
|
||||||
|
background-color: #1a1a1a;
|
||||||
|
color: #e0e0e0;
|
||||||
|
border-left: 4px solid transparent;
|
||||||
|
border-radius: 4px;
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.7);
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast .p-toast-message .p-toast-summary {
|
||||||
|
color: #ffffff;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast .p-toast-message .p-toast-detail {
|
||||||
|
color: #c0c0c0;
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast .p-toast-icon-close {
|
||||||
|
color: #ffaa00;
|
||||||
|
transition: background 0.2s;
|
||||||
|
}
|
||||||
|
.p-toast .p-toast-icon-close:hover {
|
||||||
|
background: #333;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-success {
|
||||||
|
border-left-color: #f1c40f;
|
||||||
|
}
|
||||||
|
.p-toast-message-error {
|
||||||
|
border-left-color: #e74c3c;
|
||||||
|
}
|
||||||
|
.p-toast-message-info {
|
||||||
|
border-left-color: #3498db;
|
||||||
|
}
|
||||||
|
.p-toast-message-warn {
|
||||||
|
border-left-color: #e67e22;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-success .p-toast-message-icon {
|
||||||
|
color: #f1c40f;
|
||||||
|
}
|
||||||
|
.p-toast-message-error .p-toast-message-icon {
|
||||||
|
color: #e74c3c;
|
||||||
|
}
|
||||||
|
.p-toast-message-info .p-toast-message-icon {
|
||||||
|
color: #3498db;
|
||||||
|
}
|
||||||
|
.p-toast-message-warn .p-toast-message-icon {
|
||||||
|
color: #e67e22;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-success .p-toast-message-content {
|
||||||
|
border-left-color: #f1c40f;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-error .p-toast-message-content {
|
||||||
|
border-left-color: #e74c3c;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-info .p-toast-message-content {
|
||||||
|
border-left-color: #3498db;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-warn .p-toast-message-content {
|
||||||
|
border-left-color: #e67e22;
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,18 +1,13 @@
|
|||||||
// import './tailwind.css';
|
// import './tailwind.css';
|
||||||
//@import 'primereact/resources/themes/bootstrap4-dark-blue/theme.css';
|
@use 'primereact/resources/primereact.min.css';
|
||||||
//@import 'primereact/resources/themes/lara-dark-purple/theme.css';
|
@use 'primeicons/primeicons.css';
|
||||||
//@import "prime-fixes";
|
|
||||||
@import 'primereact/resources/primereact.min.css';
|
|
||||||
//@import 'primeflex/primeflex.css';
|
|
||||||
@import 'primeicons/primeicons.css';
|
|
||||||
//@import 'primereact/resources/primereact.css';
|
|
||||||
|
|
||||||
|
|
||||||
@import "fixes";
|
@use "fixes";
|
||||||
@import "prime-fixes";
|
@use "prime-fixes";
|
||||||
@import "custom-scrollbar";
|
@use "custom-scrollbar";
|
||||||
@import "tooltip";
|
@use "tooltip";
|
||||||
@import "context-menu";
|
@use "context-menu";
|
||||||
|
|
||||||
|
|
||||||
.fixedImportant {
|
.fixedImportant {
|
||||||
|
|||||||
@@ -1,11 +1,8 @@
|
|||||||
/* Основной класс диалога */
|
|
||||||
body .p-dialog {
|
body .p-dialog {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
//position: absolute;
|
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
//visibility: hidden;
|
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
border-radius: 2px;
|
border-radius: 2px;
|
||||||
box-shadow: 0 2px 10px 0 rgba(0,0,0,0.2);
|
box-shadow: 0 2px 10px 0 rgba(0,0,0,0.2);
|
||||||
@@ -29,12 +26,10 @@ body .p-dialog {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Стиль видимого диалога */
|
|
||||||
.p-dialog-visible {
|
.p-dialog-visible {
|
||||||
visibility: visible;
|
visibility: visible;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Анимации */
|
|
||||||
.p-dialog-enter {
|
.p-dialog-enter {
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
}
|
}
|
||||||
@@ -53,31 +48,27 @@ body .p-dialog {
|
|||||||
transition: opacity 0.3s;
|
transition: opacity 0.3s;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Заголовок диалога */
|
|
||||||
.p-dialog-header {
|
.p-dialog-header {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
background: #f4f4f4;
|
background: #f4f4f4;
|
||||||
//border-bottom: 1px solid #ddd;
|
height: 40px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Содержимое диалога */
|
|
||||||
.p-dialog-content {
|
.p-dialog-content {
|
||||||
padding: 0.5rem;
|
padding: 0.5rem;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
flex: 1;
|
flex: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Подвал диалога */
|
|
||||||
.p-dialog-footer {
|
.p-dialog-footer {
|
||||||
padding: 1rem;
|
padding: .75rem 1rem;
|
||||||
border-top: 1px solid #ddd;
|
border-top: none !important;
|
||||||
background: #f4f4f4;
|
//background: #f4f4f4;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Кнопка закрытия диалога */
|
|
||||||
.p-dialog-header-close {
|
.p-dialog-header-close {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -93,3 +84,12 @@ body .p-dialog {
|
|||||||
.p-dialog-header-close .pi {
|
.p-dialog-header-close .pi {
|
||||||
font-size: 1.25rem;
|
font-size: 1.25rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.p-dialog {
|
||||||
|
.p-dialog-title {
|
||||||
|
font-size: 1rem !important;
|
||||||
|
}
|
||||||
|
.p-dialog-header-icons {
|
||||||
|
align-self: initial !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -0,0 +1,45 @@
|
|||||||
|
.p-confirm-popup {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 6px;
|
||||||
|
@apply p-[12px];
|
||||||
|
|
||||||
|
&::before, &::after {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-confirm-popup-content, .p-confirm-popup-footer {
|
||||||
|
@apply p-0 m-0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-confirm-popup-content {
|
||||||
|
display: flex;
|
||||||
|
gap: 6px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-confirm-popup-footer {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
gap: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-confirm-popup-icon {
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-confirm-popup-message {
|
||||||
|
@apply m-0;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-confirm-popup-reject.p-button-sm,
|
||||||
|
.p-confirm-popup-accept.p-button-sm {
|
||||||
|
@apply px-1.5 py-1 m-0;
|
||||||
|
|
||||||
|
& > span {
|
||||||
|
font-size: 12px;
|
||||||
|
line-height: 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,99 @@
|
|||||||
|
.vertical-tabs-container {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
min-height: 400px;
|
||||||
|
|
||||||
|
.p-tabview {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-tabview-panels {
|
||||||
|
padding: 6px 1rem;
|
||||||
|
flex-grow: 1;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-tabview-nav-container {
|
||||||
|
border-right: none;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-tabview-nav {
|
||||||
|
flex-direction: column;
|
||||||
|
width: 150px;
|
||||||
|
min-height: 100%;
|
||||||
|
border: none;
|
||||||
|
|
||||||
|
li {
|
||||||
|
width: 100%;
|
||||||
|
border-right: 4px solid var(--surface-hover);
|
||||||
|
background-color: var(--surface-card);
|
||||||
|
|
||||||
|
transition: background-color 200ms, border-right-color 200ms;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-color: var(--surface-hover);
|
||||||
|
border-right: 4px solid var(--surface-100);
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-tabview-nav-link {
|
||||||
|
transition: color 200ms;
|
||||||
|
|
||||||
|
justify-content: flex-end;
|
||||||
|
padding: 10px;
|
||||||
|
//background-color: var(--surface-card);
|
||||||
|
background-color: initial;
|
||||||
|
border: none;
|
||||||
|
color: var(--gray-400);
|
||||||
|
|
||||||
|
border-radius: initial;
|
||||||
|
font-weight: 400;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.p-tabview-selected {
|
||||||
|
background-color: var(--surface-50);
|
||||||
|
border-right: 4px solid var(--primary-color);
|
||||||
|
|
||||||
|
.p-tabview-nav-link {
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--primary-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
//background-color: var(--surface-hover);
|
||||||
|
border-right: 4px solid var(--primary-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
&.color-warn {
|
||||||
|
@apply bg-yellow-600/5 border-r-yellow-600/20;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
@apply bg-yellow-600/10 border-r-yellow-600/40;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
&.p-tabview-selected {
|
||||||
|
@apply bg-yellow-600/10 border-r-yellow-600;
|
||||||
|
|
||||||
|
.p-tabview-nav-link {
|
||||||
|
@apply text-yellow-600;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
@apply bg-yellow-600/10 border-r-yellow-600;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-tabview-panel {
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
@import "fix-dialog";
|
@use "fix-dialog";
|
||||||
//@import "fix-input";
|
@use "fix-popup";
|
||||||
|
@use "fix-tabs";
|
||||||
//@import "theme";
|
|
||||||
|
|||||||
@@ -0,0 +1,18 @@
|
|||||||
|
.Docked {
|
||||||
|
content: " ";
|
||||||
|
display: inline-block;
|
||||||
|
width: 11px;
|
||||||
|
height: 11px;
|
||||||
|
background-size: contain;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-position: center;
|
||||||
|
position: absolute;
|
||||||
|
z-index: 1;
|
||||||
|
overflow: hidden;
|
||||||
|
border-radius: 1px;
|
||||||
|
|
||||||
|
background-image: url(/images/citadelLarge.png);
|
||||||
|
left: 2px;
|
||||||
|
top: 22px;
|
||||||
|
transform: rotateZ(0deg);
|
||||||
|
}
|
||||||
@@ -1,17 +1,37 @@
|
|||||||
import { useCallback } from 'react';
|
|
||||||
import clsx from 'clsx';
|
|
||||||
import { useAutoAnimate } from '@formkit/auto-animate/react';
|
|
||||||
import { Commands } from '@/hooks/Mapper/types/mapHandlers.ts';
|
|
||||||
import { CharacterTypeRaw } from '@/hooks/Mapper/types';
|
|
||||||
import { emitMapEvent } from '@/hooks/Mapper/events';
|
import { emitMapEvent } from '@/hooks/Mapper/events';
|
||||||
|
import { isDocked } from '@/hooks/Mapper/helpers/isDocked.ts';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { CharacterTypeRaw } from '@/hooks/Mapper/types';
|
||||||
|
import { Commands, OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
import { useAutoAnimate } from '@formkit/auto-animate/react';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { useCallback } from 'react';
|
||||||
|
import classes from './Characters.module.scss';
|
||||||
|
interface CharactersProps {
|
||||||
|
data: CharacterTypeRaw[];
|
||||||
|
}
|
||||||
|
|
||||||
const Characters = ({ data }: { data: CharacterTypeRaw[] }) => {
|
export const Characters = ({ data }: CharactersProps) => {
|
||||||
const [parent] = useAutoAnimate();
|
const [parent] = useAutoAnimate();
|
||||||
|
|
||||||
const handleSelect = useCallback((character: CharacterTypeRaw) => {
|
const {
|
||||||
|
outCommand,
|
||||||
|
data: { mainCharacterEveId, followingCharacterEveId },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const handleSelect = useCallback(async (character: CharacterTypeRaw) => {
|
||||||
|
if (!character) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await outCommand({
|
||||||
|
type: OutCommand.startTracking,
|
||||||
|
data: { character_eve_id: character.eve_id },
|
||||||
|
});
|
||||||
emitMapEvent({
|
emitMapEvent({
|
||||||
name: Commands.centerSystem,
|
name: Commands.centerSystem,
|
||||||
data: character?.location?.solar_system_id?.toString(),
|
data: character.location?.solar_system_id?.toString(),
|
||||||
});
|
});
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
@@ -21,21 +41,71 @@ const Characters = ({ data }: { data: CharacterTypeRaw[] }) => {
|
|||||||
className="flex flex-col items-center justify-center"
|
className="flex flex-col items-center justify-center"
|
||||||
onClick={() => handleSelect(character)}
|
onClick={() => handleSelect(character)}
|
||||||
>
|
>
|
||||||
<div className="tooltip tooltip-bottom" title={character.name}>
|
<div
|
||||||
<a
|
className={clsx(
|
||||||
className={clsx('wd-characters-icons wd-bg-default', { ['character-online']: character.online })}
|
'overflow-hidden relative',
|
||||||
|
'flex w-[35px] h-[35px] rounded-[4px] border-[1px] border-solid bg-transparent cursor-pointer',
|
||||||
|
'transition-colors duration-250 hover:bg-stone-300/90',
|
||||||
|
{
|
||||||
|
['border-stone-800/90']: !character.online,
|
||||||
|
['border-lime-600/70']: character.online,
|
||||||
|
},
|
||||||
|
)}
|
||||||
|
title={character.tracking_paused ? `${character.name} - Tracking Paused (click to resume)` : character.name}
|
||||||
|
>
|
||||||
|
{character.tracking_paused && (
|
||||||
|
<>
|
||||||
|
<span
|
||||||
|
className={clsx(
|
||||||
|
'absolute flex flex-col p-[2px] top-[0px] left-[0px] w-[35px] h-[35px]',
|
||||||
|
'text-yellow-500 text-[9px] z-10 bg-gray-800/40',
|
||||||
|
'pi',
|
||||||
|
PrimeIcons.PAUSE,
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{mainCharacterEveId === character.eve_id && (
|
||||||
|
<span
|
||||||
|
className={clsx(
|
||||||
|
'absolute top-[2px] left-[22px] w-[9px] h-[9px]',
|
||||||
|
'text-yellow-500 text-[9px] rounded-[1px] z-10',
|
||||||
|
'pi',
|
||||||
|
PrimeIcons.STAR_FILL,
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{followingCharacterEveId === character.eve_id && (
|
||||||
|
<span
|
||||||
|
className={clsx(
|
||||||
|
'absolute top-[23px] left-[22px] w-[10px] h-[10px]',
|
||||||
|
'text-sky-300 text-[10px] rounded-[1px] z-10',
|
||||||
|
'pi pi-angle-double-right',
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{isDocked(character.location) && <div className={classes.Docked} />}
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
'flex w-full h-full bg-transparent cursor-pointer',
|
||||||
|
'bg-center bg-no-repeat bg-[length:100%]',
|
||||||
|
'transition-opacity',
|
||||||
|
'shadow-[inset_0_1px_6px_1px_#000000]',
|
||||||
|
{
|
||||||
|
['opacity-60']: !character.online,
|
||||||
|
['opacity-100']: character.online,
|
||||||
|
},
|
||||||
|
)}
|
||||||
style={{ backgroundImage: `url(https://images.evetech.net/characters/${character.eve_id}/portrait)` }}
|
style={{ backgroundImage: `url(https://images.evetech.net/characters/${character.eve_id}/portrait)` }}
|
||||||
></a>
|
></div>
|
||||||
</div>
|
</div>
|
||||||
</li>
|
</li>
|
||||||
));
|
));
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ul className="flex characters" id="characters" ref={parent}>
|
<ul className="flex gap-1 characters" id="characters" ref={parent}>
|
||||||
{items}
|
{items}
|
||||||
</ul>
|
</ul>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
// eslint-disable-next-line react/display-name
|
|
||||||
export default Characters;
|
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
import React, { RefObject } from 'react';
|
import React, { RefObject } from 'react';
|
||||||
import { ContextMenu } from 'primereact/contextmenu';
|
import { ContextMenu } from 'primereact/contextmenu';
|
||||||
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
import { PingType, SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
import { useContextMenuSystemItems } from '@/hooks/Mapper/components/contexts/ContextMenuSystem/useContextMenuSystemItems.tsx';
|
import { useContextMenuSystemItems } from '@/hooks/Mapper/components/contexts/ContextMenuSystem/useContextMenuSystemItems.tsx';
|
||||||
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
|
|
||||||
export interface ContextMenuSystemProps {
|
export interface ContextMenuSystemProps {
|
||||||
hubs: string[];
|
hubs: string[];
|
||||||
|
userHubs: string[];
|
||||||
contextMenuRef: RefObject<ContextMenu>;
|
contextMenuRef: RefObject<ContextMenu>;
|
||||||
systemId: string | undefined;
|
systemId: string | undefined;
|
||||||
systems: SolarSystemRawType[];
|
systems: SolarSystemRawType[];
|
||||||
@@ -13,10 +14,12 @@ export interface ContextMenuSystemProps {
|
|||||||
onLockToggle(): void;
|
onLockToggle(): void;
|
||||||
onOpenSettings(): void;
|
onOpenSettings(): void;
|
||||||
onHubToggle(): void;
|
onHubToggle(): void;
|
||||||
|
onUserHubToggle(): void;
|
||||||
onSystemTag(val?: string): void;
|
onSystemTag(val?: string): void;
|
||||||
onSystemStatus(val: number): void;
|
onSystemStatus(val: number): void;
|
||||||
onSystemLabels(val: string): void;
|
onSystemLabels(val: string): void;
|
||||||
onCustomLabelDialog(): void;
|
onCustomLabelDialog(): void;
|
||||||
|
onTogglePing(type: PingType, solar_system_id: string, ping_id: string | undefined, hasPing: boolean): void;
|
||||||
onWaypointSet: WaypointSetContextHandler;
|
onWaypointSet: WaypointSetContextHandler;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -25,7 +28,7 @@ export const ContextMenuSystem: React.FC<ContextMenuSystemProps> = ({ contextMen
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ContextMenu model={items} ref={contextMenuRef} breakpoint="767px" />
|
<ContextMenu className="min-w-[200px]" model={items} ref={contextMenuRef} breakpoint="767px" />
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
export * from './useTagMenu';
|
export * from './useTagMenu';
|
||||||
export * from './useStatusMenu';
|
export * from './useStatusMenu';
|
||||||
export * from './useLabelsMenu';
|
export * from './useLabelsMenu';
|
||||||
|
export * from './useUserRoute';
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
export * from './useTagMenu.ts';
|
export * from './useTagMenu.tsx';
|
||||||
|
|||||||
@@ -1,68 +0,0 @@
|
|||||||
import { MenuItem } from 'primereact/menuitem';
|
|
||||||
import { PrimeIcons } from 'primereact/api';
|
|
||||||
import { useCallback, useRef } from 'react';
|
|
||||||
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
|
||||||
import { getSystemById } from '@/hooks/Mapper/helpers';
|
|
||||||
import clsx from 'clsx';
|
|
||||||
import { GRADIENT_MENU_ACTIVE_CLASSES } from '@/hooks/Mapper/constants.ts';
|
|
||||||
|
|
||||||
const AVAILABLE_LETTERS = ['A', 'B', 'C', 'D', 'E', 'F', 'X', 'Y', 'Z'];
|
|
||||||
const AVAILABLE_NUMBERS = ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9'];
|
|
||||||
|
|
||||||
export const useTagMenu = (
|
|
||||||
systems: SolarSystemRawType[],
|
|
||||||
systemId: string | undefined,
|
|
||||||
onSystemTag: (val?: string) => void,
|
|
||||||
): (() => MenuItem) => {
|
|
||||||
const ref = useRef({ onSystemTag, systems, systemId });
|
|
||||||
ref.current = { onSystemTag, systems, systemId };
|
|
||||||
|
|
||||||
return useCallback(() => {
|
|
||||||
const { onSystemTag, systemId, systems } = ref.current;
|
|
||||||
const system = systemId ? getSystemById(systems, systemId) : undefined;
|
|
||||||
|
|
||||||
const isSelectedLetters = AVAILABLE_LETTERS.includes(system?.tag ?? '');
|
|
||||||
const isSelectedNumbers = AVAILABLE_NUMBERS.includes(system?.tag ?? '');
|
|
||||||
|
|
||||||
const menuItem: MenuItem = {
|
|
||||||
label: 'Tag',
|
|
||||||
icon: PrimeIcons.HASHTAG,
|
|
||||||
className: clsx({ [GRADIENT_MENU_ACTIVE_CLASSES]: isSelectedLetters || isSelectedNumbers }),
|
|
||||||
items: [
|
|
||||||
...(system?.tag !== '' && system?.tag !== null
|
|
||||||
? [
|
|
||||||
{
|
|
||||||
label: 'Clear',
|
|
||||||
icon: PrimeIcons.BAN,
|
|
||||||
command: () => onSystemTag(),
|
|
||||||
},
|
|
||||||
]
|
|
||||||
: []),
|
|
||||||
{
|
|
||||||
label: 'Letter',
|
|
||||||
icon: PrimeIcons.TAGS,
|
|
||||||
className: clsx({ [GRADIENT_MENU_ACTIVE_CLASSES]: isSelectedLetters }),
|
|
||||||
items: AVAILABLE_LETTERS.map(x => ({
|
|
||||||
label: x,
|
|
||||||
icon: PrimeIcons.TAG,
|
|
||||||
command: () => onSystemTag(x),
|
|
||||||
className: clsx({ [GRADIENT_MENU_ACTIVE_CLASSES]: system?.tag === x }),
|
|
||||||
})),
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Digit',
|
|
||||||
icon: PrimeIcons.TAGS,
|
|
||||||
className: clsx({ [GRADIENT_MENU_ACTIVE_CLASSES]: isSelectedNumbers }),
|
|
||||||
items: AVAILABLE_NUMBERS.map(x => ({
|
|
||||||
label: x,
|
|
||||||
icon: PrimeIcons.TAG,
|
|
||||||
command: () => onSystemTag(x),
|
|
||||||
className: clsx({ [GRADIENT_MENU_ACTIVE_CLASSES]: system?.tag === x }),
|
|
||||||
})),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
|
||||||
|
|
||||||
return menuItem;
|
|
||||||
}, []);
|
|
||||||
};
|
|
||||||
@@ -0,0 +1,95 @@
|
|||||||
|
import { MenuItem } from 'primereact/menuitem';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
|
import { getSystemById } from '@/hooks/Mapper/helpers';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { GRADIENT_MENU_ACTIVE_CLASSES } from '@/hooks/Mapper/constants.ts';
|
||||||
|
import { LayoutEventBlocker } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
|
||||||
|
const AVAILABLE_TAGS = [
|
||||||
|
'A',
|
||||||
|
'B',
|
||||||
|
'C',
|
||||||
|
'D',
|
||||||
|
'E',
|
||||||
|
'F',
|
||||||
|
'G',
|
||||||
|
'H',
|
||||||
|
'I',
|
||||||
|
'X',
|
||||||
|
'Y',
|
||||||
|
'Z',
|
||||||
|
'0',
|
||||||
|
'1',
|
||||||
|
'2',
|
||||||
|
'3',
|
||||||
|
'4',
|
||||||
|
'5',
|
||||||
|
'6',
|
||||||
|
'7',
|
||||||
|
'8',
|
||||||
|
'9',
|
||||||
|
];
|
||||||
|
|
||||||
|
export const useTagMenu = (
|
||||||
|
systems: SolarSystemRawType[],
|
||||||
|
systemId: string | undefined,
|
||||||
|
onSystemTag: (val?: string) => void,
|
||||||
|
): (() => MenuItem) => {
|
||||||
|
const ref = useRef({ onSystemTag, systems, systemId });
|
||||||
|
ref.current = { onSystemTag, systems, systemId };
|
||||||
|
|
||||||
|
return useCallback(() => {
|
||||||
|
const { onSystemTag, systemId, systems } = ref.current;
|
||||||
|
const system = systemId ? getSystemById(systems, systemId) : undefined;
|
||||||
|
|
||||||
|
const isSelectedTag = AVAILABLE_TAGS.includes(system?.tag ?? '');
|
||||||
|
|
||||||
|
const menuItem: MenuItem = {
|
||||||
|
label: 'Tag',
|
||||||
|
icon: PrimeIcons.HASHTAG,
|
||||||
|
className: clsx({ [GRADIENT_MENU_ACTIVE_CLASSES]: isSelectedTag }),
|
||||||
|
items: [
|
||||||
|
{
|
||||||
|
label: 'Digit',
|
||||||
|
icon: PrimeIcons.TAGS,
|
||||||
|
className: '!h-[128px] suppress-menu-behaviour',
|
||||||
|
template: () => {
|
||||||
|
return (
|
||||||
|
<LayoutEventBlocker className="flex flex-col gap-1 w-[200px] h-full px-2">
|
||||||
|
<div className="grid grid-cols-[auto_auto_auto_auto_auto_auto] gap-1">
|
||||||
|
{AVAILABLE_TAGS.map(x => (
|
||||||
|
<Button
|
||||||
|
outlined={system?.tag !== x}
|
||||||
|
severity="warning"
|
||||||
|
key={x}
|
||||||
|
value={x}
|
||||||
|
size="small"
|
||||||
|
className="p-[3px] justify-center"
|
||||||
|
onClick={() => system?.tag !== x && onSystemTag(x)}
|
||||||
|
>
|
||||||
|
{x}
|
||||||
|
</Button>
|
||||||
|
))}
|
||||||
|
<Button
|
||||||
|
disabled={!isSelectedTag}
|
||||||
|
icon="pi pi-ban"
|
||||||
|
size="small"
|
||||||
|
className="!p-0 !w-[initial] justify-center"
|
||||||
|
outlined
|
||||||
|
severity="help"
|
||||||
|
onClick={() => onSystemTag()}
|
||||||
|
></Button>
|
||||||
|
</div>
|
||||||
|
</LayoutEventBlocker>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
};
|
||||||
|
|
||||||
|
return menuItem;
|
||||||
|
}, []);
|
||||||
|
};
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
import { MapUserAddIcon, MapUserDeleteIcon } from '@/hooks/Mapper/icons';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
import { WidgetsIds } from '@/hooks/Mapper/components/mapInterface/constants.tsx';
|
||||||
|
|
||||||
|
interface UseUserRouteProps {
|
||||||
|
systemId: string | undefined;
|
||||||
|
userHubs: string[];
|
||||||
|
onUserHubToggle(): void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useUserRoute = ({ userHubs, systemId, onUserHubToggle }: UseUserRouteProps) => {
|
||||||
|
const {
|
||||||
|
data: { isSubscriptionActive },
|
||||||
|
windowsSettings,
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const ref = useRef({ userHubs, systemId, onUserHubToggle, isSubscriptionActive, windowsSettings });
|
||||||
|
ref.current = { userHubs, systemId, onUserHubToggle, isSubscriptionActive, windowsSettings };
|
||||||
|
|
||||||
|
return useCallback(() => {
|
||||||
|
const { userHubs, systemId, onUserHubToggle, isSubscriptionActive, windowsSettings } = ref.current;
|
||||||
|
|
||||||
|
const isVisibleUserRoutes = windowsSettings.visible.some(x => x === WidgetsIds.userRoutes);
|
||||||
|
|
||||||
|
if (!isSubscriptionActive || !isVisibleUserRoutes || !systemId) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
label: !userHubs.includes(systemId) ? 'Add User Route' : 'Remove User Route',
|
||||||
|
icon: !userHubs.includes(systemId) ? (
|
||||||
|
<MapUserAddIcon className="mr-1 relative left-[-2px]" />
|
||||||
|
) : (
|
||||||
|
<MapUserDeleteIcon className="mr-1 relative left-[-2px]" />
|
||||||
|
),
|
||||||
|
command: onUserHubToggle,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}, [windowsSettings]);
|
||||||
|
};
|
||||||
@@ -5,22 +5,29 @@ import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
|||||||
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
|
// import { PingType } from '@/hooks/Mapper/types/ping.ts';
|
||||||
|
|
||||||
interface UseContextMenuSystemHandlersProps {
|
interface UseContextMenuSystemHandlersProps {
|
||||||
hubs: string[];
|
hubs: string[];
|
||||||
|
userHubs: string[];
|
||||||
systems: SolarSystemRawType[];
|
systems: SolarSystemRawType[];
|
||||||
outCommand: OutCommandHandler;
|
outCommand: OutCommandHandler;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseContextMenuSystemHandlersProps) => {
|
export const useContextMenuSystemHandlers = ({
|
||||||
|
systems,
|
||||||
|
hubs,
|
||||||
|
userHubs,
|
||||||
|
outCommand,
|
||||||
|
}: UseContextMenuSystemHandlersProps) => {
|
||||||
const contextMenuRef = useRef<ContextMenu | null>(null);
|
const contextMenuRef = useRef<ContextMenu | null>(null);
|
||||||
|
|
||||||
const [system, setSystem] = useState<string>();
|
const [system, setSystem] = useState<string>();
|
||||||
|
|
||||||
const { deleteSystems } = useDeleteSystems();
|
const { deleteSystems } = useDeleteSystems();
|
||||||
|
|
||||||
const ref = useRef({ hubs, system, systems, outCommand, deleteSystems });
|
const ref = useRef({ hubs, userHubs, system, systems, outCommand, deleteSystems });
|
||||||
ref.current = { hubs, system, systems, outCommand, deleteSystems };
|
ref.current = { hubs, userHubs, system, systems, outCommand, deleteSystems };
|
||||||
|
|
||||||
const open = useCallback((ev: any, systemId: string) => {
|
const open = useCallback((ev: any, systemId: string) => {
|
||||||
setSystem(systemId);
|
setSystem(systemId);
|
||||||
@@ -72,6 +79,37 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
|
|||||||
setSystem(undefined);
|
setSystem(undefined);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const onUserHubToggle = useCallback(() => {
|
||||||
|
const { userHubs, system, outCommand } = ref.current;
|
||||||
|
if (!system) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
outCommand({
|
||||||
|
type: !userHubs.includes(system) ? OutCommand.addUserHub : OutCommand.deleteUserHub,
|
||||||
|
data: {
|
||||||
|
system_id: system,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
setSystem(undefined);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// const onTogglePingRally = useCallback(() => {
|
||||||
|
// const { userHubs, system, outCommand } = ref.current;
|
||||||
|
// if (!system) {
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// outCommand({
|
||||||
|
// type: OutCommand.openPing,
|
||||||
|
// data: {
|
||||||
|
// solar_system_id: system,
|
||||||
|
// type: PingType.Rally,
|
||||||
|
// },
|
||||||
|
// });
|
||||||
|
// setSystem(undefined);
|
||||||
|
// }, []);
|
||||||
|
|
||||||
const onSystemTag = useCallback((tag?: string) => {
|
const onSystemTag = useCallback((tag?: string) => {
|
||||||
const { system, outCommand } = ref.current;
|
const { system, outCommand } = ref.current;
|
||||||
if (!system) {
|
if (!system) {
|
||||||
@@ -104,7 +142,6 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
|
|||||||
setSystem(undefined);
|
setSystem(undefined);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
|
||||||
const onSystemStatus = useCallback((status: number) => {
|
const onSystemStatus = useCallback((status: number) => {
|
||||||
const { system, outCommand } = ref.current;
|
const { system, outCommand } = ref.current;
|
||||||
if (!system) {
|
if (!system) {
|
||||||
@@ -177,6 +214,8 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
|
|||||||
onDeleteSystem,
|
onDeleteSystem,
|
||||||
onLockToggle,
|
onLockToggle,
|
||||||
onHubToggle,
|
onHubToggle,
|
||||||
|
onUserHubToggle,
|
||||||
|
// onTogglePingRally,
|
||||||
onSystemTag,
|
onSystemTag,
|
||||||
onSystemTemporaryName,
|
onSystemTemporaryName,
|
||||||
onSystemStatus,
|
onSystemStatus,
|
||||||
|
|||||||
@@ -1,4 +1,9 @@
|
|||||||
import { useLabelsMenu, useStatusMenu, useTagMenu } from '@/hooks/Mapper/components/contexts/ContextMenuSystem/hooks';
|
import {
|
||||||
|
useLabelsMenu,
|
||||||
|
useStatusMenu,
|
||||||
|
useTagMenu,
|
||||||
|
useUserRoute,
|
||||||
|
} from '@/hooks/Mapper/components/contexts/ContextMenuSystem/hooks';
|
||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { getSystemById } from '@/hooks/Mapper/helpers';
|
import { getSystemById } from '@/hooks/Mapper/helpers';
|
||||||
import classes from './ContextMenuSystem.module.scss';
|
import classes from './ContextMenuSystem.module.scss';
|
||||||
@@ -9,11 +14,20 @@ import { FastSystemActions } from '@/hooks/Mapper/components/contexts/components
|
|||||||
import { useMapCheckPermissions } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
import { useMapCheckPermissions } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
import { UserPermission } from '@/hooks/Mapper/types/permissions.ts';
|
import { UserPermission } from '@/hooks/Mapper/types/permissions.ts';
|
||||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace.ts';
|
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace.ts';
|
||||||
|
import { getSystemStaticInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
|
||||||
|
import { MapAddIcon, MapDeleteIcon } from '@/hooks/Mapper/icons';
|
||||||
|
import { PingType } from '@/hooks/Mapper/types';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { MenuItem } from 'primereact/menuitem';
|
||||||
|
import { MenuItemWithInfo, WdMenuItem } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
export const useContextMenuSystemItems = ({
|
export const useContextMenuSystemItems = ({
|
||||||
onDeleteSystem,
|
onDeleteSystem,
|
||||||
onLockToggle,
|
onLockToggle,
|
||||||
onHubToggle,
|
onHubToggle,
|
||||||
|
onUserHubToggle,
|
||||||
|
onTogglePing,
|
||||||
onSystemTag,
|
onSystemTag,
|
||||||
onSystemStatus,
|
onSystemStatus,
|
||||||
onSystemLabels,
|
onSystemLabels,
|
||||||
@@ -22,6 +36,7 @@ export const useContextMenuSystemItems = ({
|
|||||||
onWaypointSet,
|
onWaypointSet,
|
||||||
systemId,
|
systemId,
|
||||||
hubs,
|
hubs,
|
||||||
|
userHubs,
|
||||||
systems,
|
systems,
|
||||||
}: Omit<ContextMenuSystemProps, 'contextMenuRef'>) => {
|
}: Omit<ContextMenuSystemProps, 'contextMenuRef'>) => {
|
||||||
const getTags = useTagMenu(systems, systemId, onSystemTag);
|
const getTags = useTagMenu(systems, systemId, onSystemTag);
|
||||||
@@ -29,9 +44,32 @@ export const useContextMenuSystemItems = ({
|
|||||||
const getLabels = useLabelsMenu(systems, systemId, onSystemLabels, onCustomLabelDialog);
|
const getLabels = useLabelsMenu(systems, systemId, onSystemLabels, onCustomLabelDialog);
|
||||||
const getWaypointMenu = useWaypointMenu(onWaypointSet);
|
const getWaypointMenu = useWaypointMenu(onWaypointSet);
|
||||||
const canLockSystem = useMapCheckPermissions([UserPermission.LOCK_SYSTEM]);
|
const canLockSystem = useMapCheckPermissions([UserPermission.LOCK_SYSTEM]);
|
||||||
|
const canManageSystem = useMapCheckPermissions([UserPermission.UPDATE_SYSTEM]);
|
||||||
|
const canDeleteSystem = useMapCheckPermissions([UserPermission.DELETE_SYSTEM]);
|
||||||
|
const getUserRoutes = useUserRoute({ userHubs, systemId, onUserHubToggle });
|
||||||
|
|
||||||
return useMemo(() => {
|
const {
|
||||||
|
data: { pings, isSubscriptionActive },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const ping = useMemo(() => (pings.length === 1 ? pings[0] : undefined), [pings]);
|
||||||
|
const isShowPingBtn = useMemo(() => {
|
||||||
|
if (!isSubscriptionActive) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pings.length === 0) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return pings[0].solar_system_id === systemId;
|
||||||
|
}, [isSubscriptionActive, pings, systemId]);
|
||||||
|
|
||||||
|
return useMemo((): MenuItem[] => {
|
||||||
const system = systemId ? getSystemById(systems, systemId) : undefined;
|
const system = systemId ? getSystemById(systems, systemId) : undefined;
|
||||||
|
const systemStaticInfo = getSystemStaticInfo(systemId)!;
|
||||||
|
|
||||||
|
const hasPing = ping?.solar_system_id === systemId;
|
||||||
|
|
||||||
if (!system || !systemId) {
|
if (!system || !systemId) {
|
||||||
return [];
|
return [];
|
||||||
@@ -44,9 +82,9 @@ export const useContextMenuSystemItems = ({
|
|||||||
return (
|
return (
|
||||||
<FastSystemActions
|
<FastSystemActions
|
||||||
systemId={systemId}
|
systemId={systemId}
|
||||||
systemName={system.system_static_info.solar_system_name}
|
systemName={systemStaticInfo.solar_system_name}
|
||||||
regionName={system.system_static_info.region_name}
|
regionName={systemStaticInfo.region_name}
|
||||||
isWH={isWormholeSpace(system.system_static_info.system_class)}
|
isWH={isWormholeSpace(systemStaticInfo.system_class)}
|
||||||
showEdit
|
showEdit
|
||||||
onOpenSettings={onOpenSettings}
|
onOpenSettings={onOpenSettings}
|
||||||
/>
|
/>
|
||||||
@@ -57,52 +95,106 @@ export const useContextMenuSystemItems = ({
|
|||||||
getTags(),
|
getTags(),
|
||||||
getStatus(),
|
getStatus(),
|
||||||
...getLabels(),
|
...getLabels(),
|
||||||
...getWaypointMenu(systemId, system.system_static_info.system_class),
|
...getWaypointMenu(systemId, systemStaticInfo.system_class),
|
||||||
{
|
{
|
||||||
label: !hubs.includes(systemId) ? 'Add in Routes' : 'Remove from Routes',
|
label: !hubs.includes(systemId) ? 'Add Route' : 'Remove Route',
|
||||||
icon: PrimeIcons.MAP_MARKER,
|
icon: !hubs.includes(systemId) ? (
|
||||||
|
<MapAddIcon className="mr-1 relative left-[-2px]" />
|
||||||
|
) : (
|
||||||
|
<MapDeleteIcon className="mr-1 relative left-[-2px]" />
|
||||||
|
),
|
||||||
command: onHubToggle,
|
command: onHubToggle,
|
||||||
},
|
},
|
||||||
...(system.locked
|
...getUserRoutes(),
|
||||||
? canLockSystem
|
|
||||||
? [
|
{ separator: true },
|
||||||
{
|
{
|
||||||
label: 'Unlock',
|
command: () => onTogglePing(PingType.Rally, systemId, ping?.id, hasPing),
|
||||||
icon: PrimeIcons.LOCK_OPEN,
|
disabled: !isShowPingBtn,
|
||||||
command: onLockToggle,
|
template: () => {
|
||||||
},
|
const iconClasses = clsx({
|
||||||
]
|
'pi text-cyan-400 hero-signal': !hasPing,
|
||||||
: []
|
'pi text-red-400 hero-signal-slash': hasPing,
|
||||||
: [
|
});
|
||||||
...(canLockSystem
|
|
||||||
? [
|
if (isShowPingBtn) {
|
||||||
{
|
return <WdMenuItem icon={iconClasses}>{!hasPing ? 'Ping: RALLY' : 'Cancel: RALLY'}</WdMenuItem>;
|
||||||
label: 'Lock',
|
}
|
||||||
icon: PrimeIcons.LOCK,
|
|
||||||
command: onLockToggle,
|
return (
|
||||||
},
|
<MenuItemWithInfo
|
||||||
]
|
infoTitle="Locked. Ping can be set only for one system."
|
||||||
: []),
|
infoClass="pi-lock text-stone-500 mr-[12px]"
|
||||||
|
>
|
||||||
|
<WdMenuItem disabled icon={iconClasses}>
|
||||||
|
{!hasPing ? 'Ping: RALLY' : 'Cancel: RALLY'}
|
||||||
|
</WdMenuItem>
|
||||||
|
</MenuItemWithInfo>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
...(system.locked && canLockSystem
|
||||||
|
? [
|
||||||
|
{
|
||||||
|
label: 'Unlock',
|
||||||
|
icon: PrimeIcons.LOCK_OPEN,
|
||||||
|
command: onLockToggle,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: []),
|
||||||
|
...(!system.locked && canManageSystem
|
||||||
|
? [
|
||||||
|
{
|
||||||
|
label: 'Lock',
|
||||||
|
icon: PrimeIcons.LOCK,
|
||||||
|
command: onLockToggle,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: []),
|
||||||
|
|
||||||
|
...(canDeleteSystem && !system.locked
|
||||||
|
? [
|
||||||
{ separator: true },
|
{ separator: true },
|
||||||
{
|
{
|
||||||
label: 'Delete',
|
|
||||||
icon: PrimeIcons.TRASH,
|
|
||||||
command: onDeleteSystem,
|
command: onDeleteSystem,
|
||||||
|
disabled: hasPing,
|
||||||
|
template: () => {
|
||||||
|
if (!hasPing) {
|
||||||
|
return <WdMenuItem icon="text-red-400 pi pi-trash">Delete</WdMenuItem>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MenuItemWithInfo
|
||||||
|
infoTitle="Locked. System can not be deleted until ping set."
|
||||||
|
infoClass="pi-lock text-stone-500 mr-[12px]"
|
||||||
|
>
|
||||||
|
<WdMenuItem disabled icon="text-red-400 pi pi-trash">
|
||||||
|
Delete
|
||||||
|
</WdMenuItem>
|
||||||
|
</MenuItemWithInfo>
|
||||||
|
);
|
||||||
|
},
|
||||||
},
|
},
|
||||||
]),
|
]
|
||||||
|
: []),
|
||||||
];
|
];
|
||||||
}, [
|
}, [
|
||||||
canLockSystem,
|
|
||||||
systems,
|
|
||||||
systemId,
|
systemId,
|
||||||
|
systems,
|
||||||
getTags,
|
getTags,
|
||||||
getStatus,
|
getStatus,
|
||||||
getLabels,
|
getLabels,
|
||||||
getWaypointMenu,
|
getWaypointMenu,
|
||||||
|
getUserRoutes,
|
||||||
hubs,
|
hubs,
|
||||||
onHubToggle,
|
onHubToggle,
|
||||||
onOpenSettings,
|
canLockSystem,
|
||||||
onLockToggle,
|
onLockToggle,
|
||||||
|
canDeleteSystem,
|
||||||
onDeleteSystem,
|
onDeleteSystem,
|
||||||
|
onOpenSettings,
|
||||||
|
onTogglePing,
|
||||||
|
ping,
|
||||||
|
isShowPingBtn,
|
||||||
]);
|
]);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import { FastSystemActions } from '@/hooks/Mapper/components/contexts/components
|
|||||||
import { useJumpPlannerMenu } from '@/hooks/Mapper/components/contexts/hooks';
|
import { useJumpPlannerMenu } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
import { Route } from '@/hooks/Mapper/types/routes.ts';
|
import { Route } from '@/hooks/Mapper/types/routes.ts';
|
||||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace.ts';
|
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace.ts';
|
||||||
|
import { MapAddIcon, MapDeleteIcon } from '@/hooks/Mapper/icons';
|
||||||
|
|
||||||
export interface ContextMenuSystemInfoProps {
|
export interface ContextMenuSystemInfoProps {
|
||||||
systemStatics: Map<number, SolarSystemStaticInfoRaw>;
|
systemStatics: Map<number, SolarSystemStaticInfoRaw>;
|
||||||
@@ -69,8 +70,12 @@ export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
|
|||||||
...getJumpPlannerMenu(system, routes),
|
...getJumpPlannerMenu(system, routes),
|
||||||
...getWaypointMenu(systemId, system.system_class),
|
...getWaypointMenu(systemId, system.system_class),
|
||||||
{
|
{
|
||||||
label: !hubs.includes(systemId) ? 'Add in Routes' : 'Remove from Routes',
|
label: !hubs.includes(systemId) ? 'Add Route' : 'Remove Route',
|
||||||
icon: PrimeIcons.MAP_MARKER,
|
icon: !hubs.includes(systemId) ? (
|
||||||
|
<MapAddIcon className="mr-1 relative left-[-2px]" />
|
||||||
|
) : (
|
||||||
|
<MapDeleteIcon className="mr-1 relative left-[-2px]" />
|
||||||
|
),
|
||||||
command: onHubToggle,
|
command: onHubToggle,
|
||||||
},
|
},
|
||||||
...(!systemOnMap
|
...(!systemOnMap
|
||||||
|
|||||||
@@ -1,25 +1,25 @@
|
|||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
import { useCallback, useRef, useState } from 'react';
|
import { useCallback, useRef, useState } from 'react';
|
||||||
import { ContextMenu } from 'primereact/contextmenu';
|
import { ContextMenu } from 'primereact/contextmenu';
|
||||||
import { Commands, OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { Commands, OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
|
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
|
||||||
import { emitMapEvent } from '@/hooks/Mapper/events';
|
import { emitMapEvent } from '@/hooks/Mapper/events';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useRouteProvider } from '@/hooks/Mapper/components/mapInterface/widgets/RoutesWidget/RoutesProvider.tsx';
|
||||||
|
|
||||||
interface UseContextMenuSystemHandlersProps {
|
export const useContextMenuSystemInfoHandlers = () => {
|
||||||
hubs: string[];
|
const { outCommand } = useMapRootState();
|
||||||
outCommand: OutCommandHandler;
|
const { hubs = [], toggleHubCommand } = useRouteProvider();
|
||||||
}
|
|
||||||
|
|
||||||
export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand }: UseContextMenuSystemHandlersProps) => {
|
|
||||||
const contextMenuRef = useRef<ContextMenu | null>(null);
|
const contextMenuRef = useRef<ContextMenu | null>(null);
|
||||||
|
|
||||||
const [system, setSystem] = useState<string>();
|
const [system, setSystem] = useState<string>();
|
||||||
const routeRef = useRef<(SolarSystemStaticInfoRaw | undefined)[]>([]);
|
const routeRef = useRef<(SolarSystemStaticInfoRaw | undefined)[]>([]);
|
||||||
|
|
||||||
const ref = useRef({ hubs, system, outCommand });
|
const ref = useRef({ hubs, system, outCommand, toggleHubCommand });
|
||||||
ref.current = { hubs, system, outCommand };
|
ref.current = { hubs, system, outCommand, toggleHubCommand };
|
||||||
|
|
||||||
const open = useCallback(
|
const open = useCallback(
|
||||||
(ev: React.SyntheticEvent, systemId: string, route: (SolarSystemStaticInfoRaw | undefined)[]) => {
|
(ev: React.SyntheticEvent, systemId: string, route: (SolarSystemStaticInfoRaw | undefined)[]) => {
|
||||||
@@ -33,17 +33,12 @@ export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand }: UseContex
|
|||||||
);
|
);
|
||||||
|
|
||||||
const onHubToggle = useCallback(() => {
|
const onHubToggle = useCallback(() => {
|
||||||
const { hubs, system, outCommand } = ref.current;
|
const { system } = ref.current;
|
||||||
if (!system) {
|
if (!system) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
outCommand({
|
ref.current.toggleHubCommand(system);
|
||||||
type: !hubs.includes(system) ? OutCommand.addHub : OutCommand.deleteHub,
|
|
||||||
data: {
|
|
||||||
system_id: system,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
setSystem(undefined);
|
setSystem(undefined);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
@@ -59,6 +54,8 @@ export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand }: UseContex
|
|||||||
system_id: solarSystemId,
|
system_id: solarSystemId,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// TODO add it to some queue
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
emitMapEvent({
|
emitMapEvent({
|
||||||
name: Commands.centerSystem,
|
name: Commands.centerSystem,
|
||||||
|
|||||||
@@ -1,17 +1,24 @@
|
|||||||
import { Node } from 'reactflow';
|
import { Node } from 'reactflow';
|
||||||
import { useCallback, useRef, useState } from 'react';
|
import { useCallback, useMemo, useRef, useState } from 'react';
|
||||||
import { ContextMenu } from 'primereact/contextmenu';
|
import { ContextMenu } from 'primereact/contextmenu';
|
||||||
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
export const useContextMenuSystemMultipleHandlers = () => {
|
export const useContextMenuSystemMultipleHandlers = () => {
|
||||||
|
const {
|
||||||
|
data: { pings },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
const contextMenuRef = useRef<ContextMenu | null>(null);
|
const contextMenuRef = useRef<ContextMenu | null>(null);
|
||||||
const [systems, setSystems] = useState<Node<SolarSystemRawType>[]>();
|
const [systems, setSystems] = useState<Node<SolarSystemRawType>[]>();
|
||||||
|
|
||||||
const { deleteSystems } = useDeleteSystems();
|
const { deleteSystems } = useDeleteSystems();
|
||||||
|
|
||||||
|
const ping = useMemo(() => (pings.length === 1 ? pings[0] : undefined), [pings]);
|
||||||
|
|
||||||
const handleSystemMultipleContext: NodeSelectionMouseHandler = (ev, systems_) => {
|
const handleSystemMultipleContext: NodeSelectionMouseHandler = (ev, systems_) => {
|
||||||
setSystems(systems_);
|
setSystems(systems_);
|
||||||
ev.preventDefault();
|
ev.preventDefault();
|
||||||
@@ -24,13 +31,17 @@ export const useContextMenuSystemMultipleHandlers = () => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const sysToDel = systems.filter(x => !x.data.locked).map(x => x.id);
|
const sysToDel = systems
|
||||||
|
.filter(x => !x.data.locked)
|
||||||
|
.filter(x => x.id !== ping?.solar_system_id)
|
||||||
|
.map(x => x.id);
|
||||||
|
|
||||||
if (sysToDel.length === 0) {
|
if (sysToDel.length === 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteSystems(sysToDel);
|
deleteSystems(sysToDel);
|
||||||
}, [deleteSystems, systems]);
|
}, [deleteSystems, systems, ping]);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
handleSystemMultipleContext,
|
handleSystemMultipleContext,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { useCallback, useRef } from 'react';
|
import { useCallback, useRef } from 'react';
|
||||||
import { LayoutEventBlocker, WdImageSize, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
import { LayoutEventBlocker, TooltipPosition, WdImageSize, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
||||||
import { ANOIK_ICON, DOTLAN_ICON, ZKB_ICON } from '@/hooks/Mapper/icons.ts';
|
import { ANOIK_ICON, DOTLAN_ICON, ZKB_ICON } from '@/hooks/Mapper/icons';
|
||||||
|
|
||||||
import classes from './FastSystemActions.module.scss';
|
import classes from './FastSystemActions.module.scss';
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
@@ -59,9 +59,21 @@ export const FastSystemActions = ({
|
|||||||
return (
|
return (
|
||||||
<LayoutEventBlocker className={clsx('flex px-2 gap-2 justify-between items-center h-full')}>
|
<LayoutEventBlocker className={clsx('flex px-2 gap-2 justify-between items-center h-full')}>
|
||||||
<div className={clsx('flex gap-2 items-center h-full', classes.Links)}>
|
<div className={clsx('flex gap-2 items-center h-full', classes.Links)}>
|
||||||
<WdImgButton tooltip={{ content: 'Open zkillboard' }} source={ZKB_ICON} onClick={handleOpenZKB} />
|
<WdImgButton
|
||||||
<WdImgButton tooltip={{ content: 'Open Anoikis' }} source={ANOIK_ICON} onClick={handleOpenAnoikis} />
|
tooltip={{ position: TooltipPosition.top, content: 'Open zkillboard' }}
|
||||||
<WdImgButton tooltip={{ content: 'Open Dotlan' }} source={DOTLAN_ICON} onClick={handleOpenDotlan} />
|
source={ZKB_ICON}
|
||||||
|
onClick={handleOpenZKB}
|
||||||
|
/>
|
||||||
|
<WdImgButton
|
||||||
|
tooltip={{ position: TooltipPosition.top, content: 'Open Anoikis' }}
|
||||||
|
source={ANOIK_ICON}
|
||||||
|
onClick={handleOpenAnoikis}
|
||||||
|
/>
|
||||||
|
<WdImgButton
|
||||||
|
tooltip={{ position: TooltipPosition.top, content: 'Open Dotlan' }}
|
||||||
|
source={DOTLAN_ICON}
|
||||||
|
onClick={handleOpenDotlan}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex gap-2 items-center pl-1">
|
<div className="flex gap-2 items-center pl-1">
|
||||||
@@ -69,14 +81,14 @@ export const FastSystemActions = ({
|
|||||||
textSize={WdImageSize.off}
|
textSize={WdImageSize.off}
|
||||||
className={PrimeIcons.COPY}
|
className={PrimeIcons.COPY}
|
||||||
onClick={copySystemNameToClipboard}
|
onClick={copySystemNameToClipboard}
|
||||||
tooltip={{ content: 'Copy system name' }}
|
tooltip={{ position: TooltipPosition.top, content: 'Copy system name' }}
|
||||||
/>
|
/>
|
||||||
{showEdit && (
|
{showEdit && (
|
||||||
<WdImgButton
|
<WdImgButton
|
||||||
textSize={WdImageSize.off}
|
textSize={WdImageSize.off}
|
||||||
className="pi pi-pen-to-square text-base"
|
className="pi pi-pen-to-square text-base"
|
||||||
onClick={onOpenSettings}
|
onClick={onOpenSettings}
|
||||||
tooltip={{ content: 'Edit system name and description' }}
|
tooltip={{ position: TooltipPosition.top, content: 'Edit system name and description' }}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -4,8 +4,8 @@ import { useCallback } from 'react';
|
|||||||
import { isPossibleSpace } from '@/hooks/Mapper/components/map/helpers/isKnownSpace.ts';
|
import { isPossibleSpace } from '@/hooks/Mapper/components/map/helpers/isKnownSpace.ts';
|
||||||
import { Route } from '@/hooks/Mapper/types/routes.ts';
|
import { Route } from '@/hooks/Mapper/types/routes.ts';
|
||||||
import { SolarSystemRawType, SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
|
import { SolarSystemRawType, SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
|
||||||
import { getSystemById } from '@/hooks/Mapper/helpers';
|
|
||||||
import { SOLAR_SYSTEM_CLASS_IDS } from '@/hooks/Mapper/components/map/constants.ts';
|
import { SOLAR_SYSTEM_CLASS_IDS } from '@/hooks/Mapper/components/map/constants.ts';
|
||||||
|
import { getSystemStaticInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
|
||||||
|
|
||||||
const imperialSpace = [SOLAR_SYSTEM_CLASS_IDS.hs, SOLAR_SYSTEM_CLASS_IDS.ls, SOLAR_SYSTEM_CLASS_IDS.ns];
|
const imperialSpace = [SOLAR_SYSTEM_CLASS_IDS.hs, SOLAR_SYSTEM_CLASS_IDS.ls, SOLAR_SYSTEM_CLASS_IDS.ns];
|
||||||
const criminalSpace = [SOLAR_SYSTEM_CLASS_IDS.ls, SOLAR_SYSTEM_CLASS_IDS.ns];
|
const criminalSpace = [SOLAR_SYSTEM_CLASS_IDS.ls, SOLAR_SYSTEM_CLASS_IDS.ns];
|
||||||
@@ -47,7 +47,7 @@ export const useJumpPlannerMenu = (
|
|||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
const origin = getSystemById(systems, systemIdFrom)?.system_static_info;
|
const origin = getSystemStaticInfo(systemIdFrom);
|
||||||
|
|
||||||
if (!origin) {
|
if (!origin) {
|
||||||
return [];
|
return [];
|
||||||
|
|||||||
1
assets/js/hooks/Mapper/components/helpers/index.ts
Normal file
1
assets/js/hooks/Mapper/components/helpers/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './parseMapUserSettings.ts';
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
import { MapUserSettings, SettingsWithVersion } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
|
||||||
|
export const REQUIRED_KEYS = [
|
||||||
|
'widgets',
|
||||||
|
'interface',
|
||||||
|
'onTheMap',
|
||||||
|
'routes',
|
||||||
|
'localWidget',
|
||||||
|
'signaturesWidget',
|
||||||
|
'killsWidget',
|
||||||
|
] as const;
|
||||||
|
|
||||||
|
type RequiredKeys = (typeof REQUIRED_KEYS)[number];
|
||||||
|
|
||||||
|
/** Custom error for any parsing / validation issue */
|
||||||
|
export class MapUserSettingsParseError extends Error {
|
||||||
|
constructor(msg: string) {
|
||||||
|
super(`MapUserSettings parse error: ${msg}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const isNumber = (v: unknown): v is number => typeof v === 'number' && !Number.isNaN(v);
|
||||||
|
|
||||||
|
/** Minimal check that an object matches SettingsWithVersion<*> */
|
||||||
|
const isSettingsWithVersion = (v: unknown): v is SettingsWithVersion<unknown> =>
|
||||||
|
typeof v === 'object' && v !== null && isNumber((v as any).version) && 'settings' in (v as any);
|
||||||
|
|
||||||
|
/** Ensure every required key is present */
|
||||||
|
const hasAllRequiredKeys = (v: unknown): v is Record<RequiredKeys, unknown> =>
|
||||||
|
typeof v === 'object' && v !== null && REQUIRED_KEYS.every(k => k in v);
|
||||||
|
|
||||||
|
/* ------------------------------ Main parser ------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses and validates a JSON string as `MapUserSettings`.
|
||||||
|
*
|
||||||
|
* @throws `MapUserSettingsParseError` – если строка не JSON или нарушена структура
|
||||||
|
*/
|
||||||
|
export const parseMapUserSettings = (json: unknown): MapUserSettings => {
|
||||||
|
if (typeof json !== 'string') throw new MapUserSettingsParseError('Input must be a JSON string');
|
||||||
|
|
||||||
|
let data: unknown;
|
||||||
|
try {
|
||||||
|
data = JSON.parse(json);
|
||||||
|
} catch (e) {
|
||||||
|
throw new MapUserSettingsParseError(`Invalid JSON: ${(e as Error).message}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!hasAllRequiredKeys(data)) {
|
||||||
|
const missing = REQUIRED_KEYS.filter(k => !(k in (data as any)));
|
||||||
|
throw new MapUserSettingsParseError(`Missing top-level field(s): ${missing.join(', ')}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const key of REQUIRED_KEYS) {
|
||||||
|
if (!isSettingsWithVersion((data as any)[key])) {
|
||||||
|
throw new MapUserSettingsParseError(`"${key}" must match SettingsWithVersion<T>`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Everything passes, so cast is safe
|
||||||
|
return data as MapUserSettings;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* ------------------------------ Usage example ----------------------------- */
|
||||||
|
|
||||||
|
// const raw = fetchFromServer(); // string
|
||||||
|
// const settings = parseMapUserSettings(raw);
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
export * from './useSystemInfo';
|
export * from './useSystemInfo';
|
||||||
export * from './useGetOwnOnlineCharacters';
|
export * from './useGetOwnOnlineCharacters';
|
||||||
export * from './useElementWidth';
|
export * from './useElementWidth';
|
||||||
|
export * from './useDetectSettingsChanged';
|
||||||
|
|||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
export const useDetectSettingsChanged = () => {
|
||||||
|
const {
|
||||||
|
storedSettings: {
|
||||||
|
interfaceSettings,
|
||||||
|
settingsRoutes,
|
||||||
|
settingsLocal,
|
||||||
|
settingsSignatures,
|
||||||
|
settingsOnTheMap,
|
||||||
|
settingsKills,
|
||||||
|
},
|
||||||
|
} = useMapRootState();
|
||||||
|
const [counter, setCounter] = useState(0);
|
||||||
|
|
||||||
|
useEffect(
|
||||||
|
() => setCounter(x => x + 1),
|
||||||
|
[interfaceSettings, settingsRoutes, settingsLocal, settingsSignatures, settingsOnTheMap, settingsKills],
|
||||||
|
);
|
||||||
|
|
||||||
|
return counter;
|
||||||
|
};
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
|
import { getSystemById } from '@/hooks/Mapper/helpers';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { getSystemById } from '@/hooks/Mapper/helpers';
|
import { getSystemStaticInfo } from '../../mapRootProvider/hooks/useLoadSystemStatic';
|
||||||
import { useLoadSystemStatic } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic.ts';
|
|
||||||
|
|
||||||
interface UseSystemInfoProps {
|
interface UseSystemInfoProps {
|
||||||
systemId: string;
|
systemId: string;
|
||||||
@@ -12,14 +12,12 @@ export const useSystemInfo = ({ systemId }: UseSystemInfoProps) => {
|
|||||||
data: { systems, connections },
|
data: { systems, connections },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const { systems: systemStatics } = useLoadSystemStatic({ systems: [systemId] });
|
|
||||||
|
|
||||||
return useMemo(() => {
|
return useMemo(() => {
|
||||||
const staticInfo = systemStatics.get(parseInt(systemId));
|
const staticInfo = getSystemStaticInfo(parseInt(systemId));
|
||||||
const dynamicInfo = getSystemById(systems, systemId);
|
const dynamicInfo = getSystemById(systems, systemId);
|
||||||
|
|
||||||
if (!staticInfo || !dynamicInfo) {
|
if (!staticInfo || !dynamicInfo) {
|
||||||
throw new Error(`Error on getting system ${systemId}`);
|
return { dynamicInfo, staticInfo, leadsTo: [] };
|
||||||
}
|
}
|
||||||
|
|
||||||
const leadsTo = connections
|
const leadsTo = connections
|
||||||
@@ -29,5 +27,5 @@ export const useSystemInfo = ({ systemId }: UseSystemInfoProps) => {
|
|||||||
.filter(x => x !== systemId);
|
.filter(x => x !== systemId);
|
||||||
|
|
||||||
return { dynamicInfo, staticInfo, leadsTo };
|
return { dynamicInfo, staticInfo, leadsTo };
|
||||||
}, [systemStatics, systemId, systems, connections]);
|
}, [systemId, systems, connections]);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,8 +1,14 @@
|
|||||||
|
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
|
import { SESSION_KEY } from '@/hooks/Mapper/constants.ts';
|
||||||
|
import { PingData, SolarSystemConnection, SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
|
import { MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
|
import type { PanelPosition } from '@reactflow/core';
|
||||||
|
import clsx from 'clsx';
|
||||||
import { ForwardedRef, forwardRef, MouseEvent, useCallback, useEffect, useMemo } from 'react';
|
import { ForwardedRef, forwardRef, MouseEvent, useCallback, useEffect, useMemo } from 'react';
|
||||||
import ReactFlow, {
|
import ReactFlow, {
|
||||||
Background,
|
Background,
|
||||||
Edge,
|
Edge,
|
||||||
EdgeChange,
|
|
||||||
MiniMap,
|
MiniMap,
|
||||||
Node,
|
Node,
|
||||||
NodeChange,
|
NodeChange,
|
||||||
@@ -17,8 +23,6 @@ import ReactFlow, {
|
|||||||
import 'reactflow/dist/style.css';
|
import 'reactflow/dist/style.css';
|
||||||
import classes from './Map.module.scss';
|
import classes from './Map.module.scss';
|
||||||
import { MapProvider, useMapState } from './MapProvider';
|
import { MapProvider, useMapState } from './MapProvider';
|
||||||
import { useEdgesState, useMapHandlers, useNodesState, useUpdateNodes } from './hooks';
|
|
||||||
import { MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
|
||||||
import {
|
import {
|
||||||
ContextMenuConnection,
|
ContextMenuConnection,
|
||||||
ContextMenuRoot,
|
ContextMenuRoot,
|
||||||
@@ -27,13 +31,9 @@ import {
|
|||||||
useContextMenuRootHandlers,
|
useContextMenuRootHandlers,
|
||||||
} from './components';
|
} from './components';
|
||||||
import { getBehaviorForTheme } from './helpers/getThemeBehavior';
|
import { getBehaviorForTheme } from './helpers/getThemeBehavior';
|
||||||
import { OnMapAddSystemCallback, OnMapSelectionChange } from './map.types';
|
import { useEdgesState, useMapHandlers, useNodesState, useUpdateNodes } from './hooks';
|
||||||
import { SESSION_KEY } from '@/hooks/Mapper/constants.ts';
|
|
||||||
import { SolarSystemConnection, SolarSystemRawType } from '@/hooks/Mapper/types';
|
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
|
||||||
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
|
||||||
import clsx from 'clsx';
|
|
||||||
import { useBackgroundVars } from './hooks/useBackgroundVars';
|
import { useBackgroundVars } from './hooks/useBackgroundVars';
|
||||||
|
import { OnMapAddSystemCallback, OnMapSelectionChange } from './map.types';
|
||||||
|
|
||||||
const DEFAULT_VIEW_PORT = { zoom: 1, x: 0, y: 0 };
|
const DEFAULT_VIEW_PORT = { zoom: 1, x: 0, y: 0 };
|
||||||
|
|
||||||
@@ -79,11 +79,12 @@ const edgeTypes = {
|
|||||||
floating: SolarSystemEdge,
|
floating: SolarSystemEdge,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const MAP_ROOT_ID = 'MAP_ROOT_ID';
|
||||||
|
|
||||||
interface MapCompProps {
|
interface MapCompProps {
|
||||||
refn: ForwardedRef<MapHandlers>;
|
refn: ForwardedRef<MapHandlers>;
|
||||||
onCommand: OutCommandHandler;
|
onCommand: OutCommandHandler;
|
||||||
onSelectionChange: OnMapSelectionChange;
|
onSelectionChange: OnMapSelectionChange;
|
||||||
onManualDelete(systems: string[]): void;
|
|
||||||
onConnectionInfoClick?(e: SolarSystemConnection): void;
|
onConnectionInfoClick?(e: SolarSystemConnection): void;
|
||||||
onAddSystem?: OnMapAddSystemCallback;
|
onAddSystem?: OnMapAddSystemCallback;
|
||||||
onSelectionContextMenu?: NodeSelectionMouseHandler;
|
onSelectionContextMenu?: NodeSelectionMouseHandler;
|
||||||
@@ -95,6 +96,9 @@ interface MapCompProps {
|
|||||||
isShowBackgroundPattern?: boolean;
|
isShowBackgroundPattern?: boolean;
|
||||||
isSoftBackground?: boolean;
|
isSoftBackground?: boolean;
|
||||||
theme?: string;
|
theme?: string;
|
||||||
|
pings: PingData[];
|
||||||
|
minimapPlacement?: PanelPosition;
|
||||||
|
localShowShipName?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const MapComp = ({
|
const MapComp = ({
|
||||||
@@ -105,7 +109,6 @@ const MapComp = ({
|
|||||||
onSystemContextMenu,
|
onSystemContextMenu,
|
||||||
onConnectionInfoClick,
|
onConnectionInfoClick,
|
||||||
onSelectionContextMenu,
|
onSelectionContextMenu,
|
||||||
onManualDelete,
|
|
||||||
isShowMinimap,
|
isShowMinimap,
|
||||||
showKSpaceBG,
|
showKSpaceBG,
|
||||||
isThickConnections,
|
isThickConnections,
|
||||||
@@ -113,8 +116,11 @@ const MapComp = ({
|
|||||||
isSoftBackground,
|
isSoftBackground,
|
||||||
theme,
|
theme,
|
||||||
onAddSystem,
|
onAddSystem,
|
||||||
|
pings,
|
||||||
|
minimapPlacement = 'bottom-right',
|
||||||
|
localShowShipName = false,
|
||||||
}: MapCompProps) => {
|
}: MapCompProps) => {
|
||||||
const { getNode, getNodes } = useReactFlow();
|
const { getNodes } = useReactFlow();
|
||||||
const [nodes, , onNodesChange] = useNodesState<Node<SolarSystemRawType>>(initialNodes);
|
const [nodes, , onNodesChange] = useNodesState<Node<SolarSystemRawType>>(initialNodes);
|
||||||
const [edges, , onEdgesChange] = useEdgesState<Edge<SolarSystemConnection>>(initialEdges);
|
const [edges, , onEdgesChange] = useEdgesState<Edge<SolarSystemConnection>>(initialEdges);
|
||||||
|
|
||||||
@@ -187,8 +193,6 @@ const MapComp = ({
|
|||||||
|
|
||||||
const handleNodesChange = useCallback(
|
const handleNodesChange = useCallback(
|
||||||
(changes: NodeChange[]) => {
|
(changes: NodeChange[]) => {
|
||||||
const systemsIdsToRemove: string[] = [];
|
|
||||||
|
|
||||||
// prevents single node deselection on background / same node click
|
// prevents single node deselection on background / same node click
|
||||||
// allows deseletion of all nodes if multiple are currently selected
|
// allows deseletion of all nodes if multiple are currently selected
|
||||||
if (changes.length === 1 && changes[0].type == 'select' && changes[0].selected === false) {
|
if (changes.length === 1 && changes[0].type == 'select' && changes[0].selected === false) {
|
||||||
@@ -196,30 +200,12 @@ const MapComp = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const nextChanges = changes.reduce((acc, change) => {
|
const nextChanges = changes.reduce((acc, change) => {
|
||||||
if (change.type !== 'remove') {
|
|
||||||
return [...acc, change];
|
|
||||||
}
|
|
||||||
|
|
||||||
const node = getNode(change.id);
|
|
||||||
if (!node) {
|
|
||||||
return [...acc, change];
|
|
||||||
}
|
|
||||||
|
|
||||||
if (node.data.locked) {
|
|
||||||
return acc;
|
|
||||||
}
|
|
||||||
|
|
||||||
systemsIdsToRemove.push(node.data.id);
|
|
||||||
return [...acc, change];
|
return [...acc, change];
|
||||||
}, [] as NodeChange[]);
|
}, [] as NodeChange[]);
|
||||||
|
|
||||||
if (systemsIdsToRemove.length > 0) {
|
|
||||||
onManualDelete(systemsIdsToRemove);
|
|
||||||
}
|
|
||||||
|
|
||||||
onNodesChange(nextChanges);
|
onNodesChange(nextChanges);
|
||||||
},
|
},
|
||||||
[getNode, getNodes, onManualDelete, onNodesChange],
|
[getNodes, onNodesChange],
|
||||||
);
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -227,12 +213,17 @@ const MapComp = ({
|
|||||||
...x,
|
...x,
|
||||||
showKSpaceBG: showKSpaceBG,
|
showKSpaceBG: showKSpaceBG,
|
||||||
isThickConnections: isThickConnections,
|
isThickConnections: isThickConnections,
|
||||||
|
pings,
|
||||||
|
localShowShipName,
|
||||||
}));
|
}));
|
||||||
}, [showKSpaceBG, isThickConnections, update]);
|
}, [showKSpaceBG, isThickConnections, pings, update, localShowShipName]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className={clsx(classes.MapRoot, { [classes.BackgroundAlternateColor]: isSoftBackground })}>
|
<div
|
||||||
|
data-window-id={MAP_ROOT_ID}
|
||||||
|
className={clsx(classes.MapRoot, { [classes.BackgroundAlternateColor]: isSoftBackground })}
|
||||||
|
>
|
||||||
<ReactFlow
|
<ReactFlow
|
||||||
nodes={nodes}
|
nodes={nodes}
|
||||||
edges={edges}
|
edges={edges}
|
||||||
@@ -276,7 +267,7 @@ const MapComp = ({
|
|||||||
minZoom={0.2}
|
minZoom={0.2}
|
||||||
maxZoom={1.5}
|
maxZoom={1.5}
|
||||||
elevateNodesOnSelect
|
elevateNodesOnSelect
|
||||||
deleteKeyCode={['Delete']}
|
deleteKeyCode={['']}
|
||||||
{...(isPanAndDrag
|
{...(isPanAndDrag
|
||||||
? {
|
? {
|
||||||
selectionOnDrag: true,
|
selectionOnDrag: true,
|
||||||
@@ -288,7 +279,9 @@ const MapComp = ({
|
|||||||
// onlyRenderVisibleElements
|
// onlyRenderVisibleElements
|
||||||
selectionMode={SelectionMode.Partial}
|
selectionMode={SelectionMode.Partial}
|
||||||
>
|
>
|
||||||
{isShowMinimap && <MiniMap pannable zoomable ariaLabel="Mini map" className={minimapClasses} />}
|
{isShowMinimap && (
|
||||||
|
<MiniMap pannable zoomable ariaLabel="Mini map" className={minimapClasses} position={minimapPlacement} />
|
||||||
|
)}
|
||||||
{isShowBackgroundPattern && <Background variant={variant} gap={gap} size={size} color={color} />}
|
{isShowBackgroundPattern && <Background variant={variant} gap={gap} size={size} color={color} />}
|
||||||
</ReactFlow>
|
</ReactFlow>
|
||||||
{/* <button className="z-auto btn btn-primary absolute top-20 right-20" onClick={handleGetPassages}>
|
{/* <button className="z-auto btn btn-primary absolute top-20 right-20" onClick={handleGetPassages}>
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ export type MapData = MapUnionTypes & {
|
|||||||
showKSpaceBG: boolean;
|
showKSpaceBG: boolean;
|
||||||
isThickConnections: boolean;
|
isThickConnections: boolean;
|
||||||
linkedSigEveId: string;
|
linkedSigEveId: string;
|
||||||
|
localShowShipName: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
interface MapProviderProps {
|
interface MapProviderProps {
|
||||||
@@ -38,6 +39,11 @@ const INITIAL_DATA: MapData = {
|
|||||||
systemSignatures: {} as Record<string, SystemSignature[]>,
|
systemSignatures: {} as Record<string, SystemSignature[]>,
|
||||||
options: {} as Record<string, string | boolean>,
|
options: {} as Record<string, string | boolean>,
|
||||||
isSubscriptionActive: false,
|
isSubscriptionActive: false,
|
||||||
|
mainCharacterEveId: null,
|
||||||
|
followingCharacterEveId: null,
|
||||||
|
userHubs: [],
|
||||||
|
pings: [],
|
||||||
|
localShowShipName: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export interface MapContextProps {
|
export interface MapContextProps {
|
||||||
@@ -53,7 +59,7 @@ const MapContext = createContext<MapContextProps>({
|
|||||||
outCommand: async () => void 0,
|
outCommand: async () => void 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
export const MapProvider: React.FC<MapProviderProps> = ({ children, onCommand }) => {
|
export const MapProvider = ({ children, onCommand }: MapProviderProps) => {
|
||||||
const { update, ref } = useContextStore<MapData>({ ...INITIAL_DATA });
|
const { update, ref } = useContextStore<MapData>({ ...INITIAL_DATA });
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
@use '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
||||||
|
|
||||||
.ConnectionTimeEOL {
|
.ConnectionTimeEOL {
|
||||||
background-image: linear-gradient(207deg, transparent, var(--conn-time-eol));
|
background-image: linear-gradient(207deg, transparent, var(--conn-time-eol));
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import React, { RefObject, useMemo } from 'react';
|
|||||||
import { ContextMenu } from 'primereact/contextmenu';
|
import { ContextMenu } from 'primereact/contextmenu';
|
||||||
import { PrimeIcons } from 'primereact/api';
|
import { PrimeIcons } from 'primereact/api';
|
||||||
import { MenuItem } from 'primereact/menuitem';
|
import { MenuItem } from 'primereact/menuitem';
|
||||||
import { Edge } from '@reactflow/core/dist/esm/types/edges';
|
|
||||||
import { ConnectionType, MassState, ShipSizeStatus, SolarSystemConnection, TimeStatus } from '@/hooks/Mapper/types';
|
import { ConnectionType, MassState, ShipSizeStatus, SolarSystemConnection, TimeStatus } from '@/hooks/Mapper/types';
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import classes from './ContextMenuConnection.module.scss';
|
import classes from './ContextMenuConnection.module.scss';
|
||||||
@@ -14,6 +13,7 @@ import {
|
|||||||
SHIP_SIZES_NAMES_SHORT,
|
SHIP_SIZES_NAMES_SHORT,
|
||||||
SHIP_SIZES_SIZE,
|
SHIP_SIZES_SIZE,
|
||||||
} from '@/hooks/Mapper/components/map/constants.ts';
|
} from '@/hooks/Mapper/components/map/constants.ts';
|
||||||
|
import { Edge } from 'reactflow';
|
||||||
|
|
||||||
export interface ContextMenuConnectionProps {
|
export interface ContextMenuConnectionProps {
|
||||||
contextMenuRef: RefObject<ContextMenu>;
|
contextMenuRef: RefObject<ContextMenu>;
|
||||||
|
|||||||
@@ -1,9 +1,8 @@
|
|||||||
import { EdgeMouseHandler } from 'reactflow';
|
import { Edge, EdgeMouseHandler } from 'reactflow';
|
||||||
import { useCallback, useRef, useState } from 'react';
|
import { useCallback, useRef, useState } from 'react';
|
||||||
import { ContextMenu } from 'primereact/contextmenu';
|
import { ContextMenu } from 'primereact/contextmenu';
|
||||||
import { useMapState } from '../../MapProvider.tsx';
|
import { useMapState } from '../../MapProvider.tsx';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { Edge } from '@reactflow/core/dist/esm/types/edges';
|
|
||||||
import { ConnectionType, MassState, ShipSizeStatus, SolarSystemConnection, TimeStatus } from '@/hooks/Mapper/types';
|
import { ConnectionType, MassState, ShipSizeStatus, SolarSystemConnection, TimeStatus } from '@/hooks/Mapper/types';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,64 @@
|
|||||||
|
import { useMemo } from 'react';
|
||||||
|
import { useKillsCounter } from '../../hooks/useKillsCounter.ts';
|
||||||
|
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
||||||
|
import { WithChildren, WithClassName } from '@/hooks/Mapper/types/common.ts';
|
||||||
|
import {
|
||||||
|
KILLS_ROW_HEIGHT,
|
||||||
|
SystemKillsList,
|
||||||
|
} from '@/hooks/Mapper/components/mapInterface/widgets/WSystemKills/SystemKillsList';
|
||||||
|
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
|
||||||
|
|
||||||
|
const MIN_TOOLTIP_HEIGHT = 40;
|
||||||
|
|
||||||
|
type KillsBookmarkTooltipProps = {
|
||||||
|
killsCount: number;
|
||||||
|
killsActivityType: string | null;
|
||||||
|
systemId: string;
|
||||||
|
className?: string;
|
||||||
|
size?: TooltipSize;
|
||||||
|
} & WithChildren &
|
||||||
|
WithClassName;
|
||||||
|
|
||||||
|
export const KillsCounter = ({
|
||||||
|
killsCount,
|
||||||
|
systemId,
|
||||||
|
className,
|
||||||
|
children,
|
||||||
|
size = TooltipSize.xs,
|
||||||
|
}: KillsBookmarkTooltipProps) => {
|
||||||
|
const { isLoading, kills: detailedKills } = useKillsCounter({
|
||||||
|
realSystemId: systemId,
|
||||||
|
});
|
||||||
|
|
||||||
|
const limitedKills = useMemo(() => {
|
||||||
|
if (!detailedKills || detailedKills.length === 0) return [];
|
||||||
|
return detailedKills.slice(0, killsCount);
|
||||||
|
}, [detailedKills, killsCount]);
|
||||||
|
|
||||||
|
if (!killsCount || limitedKills.length === 0 || !systemId || isLoading) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate height based on number of kills, but ensure a minimum height
|
||||||
|
const killsNeededHeight = limitedKills.length * KILLS_ROW_HEIGHT;
|
||||||
|
// Add a small buffer (10px) to prevent scrollbar from appearing unnecessarily
|
||||||
|
const tooltipHeight = Math.max(MIN_TOOLTIP_HEIGHT, Math.min(killsNeededHeight + 10, 500));
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WdTooltipWrapper
|
||||||
|
content={
|
||||||
|
<div className="overflow-hidden flex w-[450px] flex-col" style={{ height: `${tooltipHeight}px` }}>
|
||||||
|
<div className="flex-1 h-full">
|
||||||
|
<SystemKillsList kills={limitedKills} onlyOneSystem timeRange={1} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
className={className}
|
||||||
|
tooltipClassName="!px-0"
|
||||||
|
size={size}
|
||||||
|
interactive={true}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './KillsCounter.tsx';
|
||||||
@@ -3,11 +3,11 @@ import clsx from 'clsx';
|
|||||||
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
||||||
import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit/WdTooltip';
|
import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit/WdTooltip';
|
||||||
import { CharItemProps, LocalCharactersList } from '../../../mapInterface/widgets/LocalCharacters/components';
|
import { CharItemProps, LocalCharactersList } from '../../../mapInterface/widgets/LocalCharacters/components';
|
||||||
import { useLocalCharactersItemTemplate } from '../../../mapInterface/widgets/LocalCharacters/hooks/useLocalCharacters';
|
|
||||||
import { useLocalCharacterWidgetSettings } from '../../../mapInterface/widgets/LocalCharacters/hooks/useLocalWidgetSettings';
|
|
||||||
import classes from './SolarSystemLocalCounter.module.scss';
|
|
||||||
import { AvailableThemes } from '@/hooks/Mapper/mapRootProvider';
|
|
||||||
import { useTheme } from '@/hooks/Mapper/hooks/useTheme.ts';
|
import { useTheme } from '@/hooks/Mapper/hooks/useTheme.ts';
|
||||||
|
import { AvailableThemes } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import classes from './LocalCounter.module.scss';
|
||||||
|
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
||||||
|
import { useLocalCharactersItemTemplate } from '@/hooks/Mapper/components/mapInterface/widgets/LocalCharacters/hooks/useLocalCharacters.tsx';
|
||||||
|
|
||||||
interface LocalCounterProps {
|
interface LocalCounterProps {
|
||||||
localCounterCharacters: Array<CharItemProps>;
|
localCounterCharacters: Array<CharItemProps>;
|
||||||
@@ -16,8 +16,10 @@ interface LocalCounterProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const LocalCounter = ({ localCounterCharacters, hasUserCharacters, showIcon = true }: LocalCounterProps) => {
|
export const LocalCounter = ({ localCounterCharacters, hasUserCharacters, showIcon = true }: LocalCounterProps) => {
|
||||||
const [settings] = useLocalCharacterWidgetSettings();
|
const {
|
||||||
const itemTemplate = useLocalCharactersItemTemplate(settings.showShipName);
|
data: { localShowShipName },
|
||||||
|
} = useMapState();
|
||||||
|
const itemTemplate = useLocalCharactersItemTemplate(localShowShipName);
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
const pilotTooltipContent = useMemo(() => {
|
const pilotTooltipContent = useMemo(() => {
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './LocalCounter';
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
@use '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
||||||
|
|
||||||
.EdgePathBack {
|
.EdgePathBack {
|
||||||
fill: none;
|
fill: none;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import classes from './SolarSystemEdge.module.scss';
|
import classes from './SolarSystemEdge.module.scss';
|
||||||
import { EdgeLabelRenderer, EdgeProps, getBezierPath, getSmoothStepPath, Position, useStore } from 'reactflow';
|
import { EdgeLabelRenderer, EdgeProps, getBezierPath, Position, useStore } from 'reactflow';
|
||||||
import { getEdgeParams } from '@/hooks/Mapper/components/map/utils.ts';
|
import { getEdgeParams } from '@/hooks/Mapper/components/map/utils.ts';
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import { ConnectionType, MassState, ShipSizeStatus, SolarSystemConnection, TimeStatus } from '@/hooks/Mapper/types';
|
import { ConnectionType, MassState, ShipSizeStatus, SolarSystemConnection, TimeStatus } from '@/hooks/Mapper/types';
|
||||||
@@ -51,11 +51,11 @@ export const SolarSystemEdge = ({ id, source, target, markerEnd, style, data }:
|
|||||||
const [hovered, setHovered] = useState(false);
|
const [hovered, setHovered] = useState(false);
|
||||||
|
|
||||||
const [path, labelX, labelY, sx, sy, tx, ty, sourcePos, targetPos] = useMemo(() => {
|
const [path, labelX, labelY, sx, sy, tx, ty, sourcePos, targetPos] = useMemo(() => {
|
||||||
const { sx, sy, tx, ty, sourcePos, targetPos } = getEdgeParams(sourceNode, targetNode);
|
const { sx, sy, tx, ty, sourcePos, targetPos } = getEdgeParams(sourceNode!, targetNode!);
|
||||||
|
|
||||||
const offset = isThickConnections ? MAP_OFFSETS_TICK[targetPos] : MAP_OFFSETS[targetPos];
|
const offset = isThickConnections ? MAP_OFFSETS_TICK[targetPos] : MAP_OFFSETS[targetPos];
|
||||||
|
|
||||||
const method = isWormhole ? getBezierPath : getSmoothStepPath;
|
const method = isWormhole ? getBezierPath : getBezierPath;
|
||||||
|
|
||||||
const [edgePath, labelX, labelY] = method({
|
const [edgePath, labelX, labelY] = method({
|
||||||
sourceX: sx - offset.x,
|
sourceX: sx - offset.x,
|
||||||
|
|||||||
@@ -1,39 +0,0 @@
|
|||||||
import { SystemKillsContent } from '../../../mapInterface/widgets/SystemKills/SystemKillsContent/SystemKillsContent';
|
|
||||||
import { useKillsCounter } from '../../hooks/useKillsCounter';
|
|
||||||
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
|
||||||
import { WithChildren, WithClassName } from '@/hooks/Mapper/types/common';
|
|
||||||
|
|
||||||
type TooltipSize = 'xs' | 'sm' | 'md' | 'lg';
|
|
||||||
|
|
||||||
type KillsBookmarkTooltipProps = {
|
|
||||||
killsCount: number;
|
|
||||||
killsActivityType: string | null;
|
|
||||||
systemId: string;
|
|
||||||
className?: string;
|
|
||||||
size?: TooltipSize;
|
|
||||||
} & WithChildren &
|
|
||||||
WithClassName;
|
|
||||||
|
|
||||||
export const KillsCounter = ({ killsCount, systemId, className, children, size = 'xs' }: KillsBookmarkTooltipProps) => {
|
|
||||||
const { isLoading, kills: detailedKills, systemNameMap } = useKillsCounter({ realSystemId: systemId });
|
|
||||||
|
|
||||||
if (!killsCount || detailedKills.length === 0 || !systemId || isLoading) return null;
|
|
||||||
|
|
||||||
const tooltipContent = (
|
|
||||||
<div style={{ width: '100%', minWidth: '300px', overflow: 'hidden' }}>
|
|
||||||
<SystemKillsContent
|
|
||||||
kills={detailedKills}
|
|
||||||
systemNameMap={systemNameMap}
|
|
||||||
onlyOneSystem={true}
|
|
||||||
autoSize={true}
|
|
||||||
limit={killsCount}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<WdTooltipWrapper content={tooltipContent} className={className} size={size} interactive={true}>
|
|
||||||
{children}
|
|
||||||
</WdTooltipWrapper>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@@ -1,11 +1,24 @@
|
|||||||
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
@use "sass:color";
|
||||||
|
@use '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
||||||
|
|
||||||
$pastel-blue: #5a7d9a;
|
$pastel-blue: #5a7d9a;
|
||||||
$pastel-pink: #d291bc;
|
$pastel-pink: rgb(30, 161, 255);
|
||||||
$dark-bg: #2d2d2d;
|
$dark-bg: #2d2d2d;
|
||||||
$text-color: #ffffff;
|
$text-color: #ffffff;
|
||||||
$tooltip-bg: #202020;
|
$tooltip-bg: #202020;
|
||||||
|
|
||||||
|
$neon-color-1: rgb(27, 132, 236);
|
||||||
|
$neon-color-3: rgba(27, 132, 236, 0.40);
|
||||||
|
|
||||||
|
@keyframes move-stripes {
|
||||||
|
from {
|
||||||
|
background-position: 0 0;
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
background-position: 30px 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.RootCustomNode {
|
.RootCustomNode {
|
||||||
display: flex;
|
display: flex;
|
||||||
width: 130px;
|
width: 130px;
|
||||||
@@ -22,17 +35,18 @@ $tooltip-bg: #202020;
|
|||||||
color: var(--rf-text-color, #ffffff);
|
color: var(--rf-text-color, #ffffff);
|
||||||
|
|
||||||
box-shadow: 0 0 5px rgba($dark-bg, 0.5);
|
box-shadow: 0 0 5px rgba($dark-bg, 0.5);
|
||||||
border: 1px solid darken($pastel-blue, 10%);
|
border: 1px solid color.adjust($pastel-blue, $lightness: -10%);
|
||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
position: relative;
|
position: relative;
|
||||||
z-index: 3;
|
z-index: 3;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
|
||||||
|
&.Pochven,
|
||||||
&.Mataria,
|
&.Mataria,
|
||||||
&.Amarria,
|
&.Amarria,
|
||||||
&.Gallente,
|
&.Gallente,
|
||||||
&.Caldaria {
|
&.Caldaria {
|
||||||
&::before {
|
&::after {
|
||||||
content: '';
|
content: '';
|
||||||
position: absolute;
|
position: absolute;
|
||||||
top: 0;
|
top: 0;
|
||||||
@@ -48,7 +62,7 @@ $tooltip-bg: #202020;
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.Mataria {
|
&.Mataria {
|
||||||
&::before {
|
&::after {
|
||||||
background-image: url('/images/mataria-180.png');
|
background-image: url('/images/mataria-180.png');
|
||||||
opacity: 0.6;
|
opacity: 0.6;
|
||||||
background-position-x: 1px;
|
background-position-x: 1px;
|
||||||
@@ -57,7 +71,7 @@ $tooltip-bg: #202020;
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.Caldaria {
|
&.Caldaria {
|
||||||
&::before {
|
&::after {
|
||||||
background-image: url('/images/caldaria-180.png');
|
background-image: url('/images/caldaria-180.png');
|
||||||
opacity: 0.6;
|
opacity: 0.6;
|
||||||
background-position-x: 1px;
|
background-position-x: 1px;
|
||||||
@@ -66,7 +80,7 @@ $tooltip-bg: #202020;
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.Amarria {
|
&.Amarria {
|
||||||
&::before {
|
&::after {
|
||||||
opacity: 0.45;
|
opacity: 0.45;
|
||||||
background-image: url('/images/amarr-180.png');
|
background-image: url('/images/amarr-180.png');
|
||||||
background-position-x: 0;
|
background-position-x: 0;
|
||||||
@@ -75,7 +89,7 @@ $tooltip-bg: #202020;
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.Gallente {
|
&.Gallente {
|
||||||
&::before {
|
&::after {
|
||||||
opacity: 0.5;
|
opacity: 0.5;
|
||||||
background-image: url('/images/gallente-180.png');
|
background-image: url('/images/gallente-180.png');
|
||||||
background-position-x: 1px;
|
background-position-x: 1px;
|
||||||
@@ -83,11 +97,43 @@ $tooltip-bg: #202020;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.Pochven {
|
||||||
|
&::after {
|
||||||
|
opacity: 0.8;
|
||||||
|
background-image: url('/images/pochven.webp');
|
||||||
|
background-position-x: 0;
|
||||||
|
background-position-y: -13px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
&.selected {
|
&.selected {
|
||||||
border-color: $pastel-pink;
|
border-color: $pastel-pink;
|
||||||
box-shadow: 0 0 10px #9a1af1c2;
|
box-shadow: 0 0 10px #9a1af1c2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.rally {
|
||||||
|
&::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
z-index: -1;
|
||||||
|
|
||||||
|
border-color: $neon-color-1;
|
||||||
|
background: repeating-linear-gradient(
|
||||||
|
45deg,
|
||||||
|
$neon-color-3 0px,
|
||||||
|
$neon-color-3 8px,
|
||||||
|
transparent 8px,
|
||||||
|
transparent 21px
|
||||||
|
);
|
||||||
|
background-size: 30px 30px;
|
||||||
|
animation: move-stripes 3s linear infinite;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
&.eve-system-status-home {
|
&.eve-system-status-home {
|
||||||
border: 1px solid var(--eve-solar-system-status-color-home-dark30);
|
border: 1px solid var(--eve-solar-system-status-color-home-dark30);
|
||||||
background-image: linear-gradient(45deg, var(--eve-solar-system-status-color-background), transparent);
|
background-image: linear-gradient(45deg, var(--eve-solar-system-status-color-background), transparent);
|
||||||
@@ -355,3 +401,15 @@ $tooltip-bg: #202020;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.ShatteredIcon {
|
||||||
|
position: relative;
|
||||||
|
//top: -1px;
|
||||||
|
left: -1px;
|
||||||
|
|
||||||
|
background-size: 100%;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-position: center;
|
||||||
|
|
||||||
|
background-image: url(/images/chart-network-svgrepo-com.svg)
|
||||||
|
}
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import { Handle, NodeProps, Position } from 'reactflow';
|
|||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import classes from './SolarSystemNodeDefault.module.scss';
|
import classes from './SolarSystemNodeDefault.module.scss';
|
||||||
import { PrimeIcons } from 'primereact/api';
|
import { PrimeIcons } from 'primereact/api';
|
||||||
import { useLocalCounter, useSolarSystemNode, useNodeKillsCount } from '../../hooks/useSolarSystemLogic';
|
import { useLocalCounter, useNodeKillsCount, useSolarSystemNode } from '../../hooks';
|
||||||
import {
|
import {
|
||||||
EFFECT_BACKGROUND_STYLES,
|
EFFECT_BACKGROUND_STYLES,
|
||||||
MARKER_BOOKMARK_BG_STYLES,
|
MARKER_BOOKMARK_BG_STYLES,
|
||||||
@@ -12,45 +12,55 @@ import {
|
|||||||
} from '@/hooks/Mapper/components/map/constants';
|
} from '@/hooks/Mapper/components/map/constants';
|
||||||
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
||||||
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
||||||
import { LocalCounter } from './SolarSystemLocalCounter';
|
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
|
||||||
import { KillsCounter } from './SolarSystemKillsCounter';
|
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { Tag } from 'primereact/tag';
|
||||||
|
import { LocalCounter } from '@/hooks/Mapper/components/map/components/LocalCounter';
|
||||||
|
import { KillsCounter } from '@/hooks/Mapper/components/map/components/KillsCounter';
|
||||||
|
|
||||||
|
// let render = 0;
|
||||||
export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>) => {
|
export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>) => {
|
||||||
const nodeVars = useSolarSystemNode(props);
|
const nodeVars = useSolarSystemNode(props);
|
||||||
const { localCounterCharacters } = useLocalCounter(nodeVars);
|
const { localCounterCharacters } = useLocalCounter(nodeVars);
|
||||||
const localKillsCount = useNodeKillsCount(nodeVars.solarSystemId, nodeVars.killsCount);
|
const { killsCount: localKillsCount, killsActivityType: localKillsActivityType } = useNodeKillsCount(
|
||||||
|
nodeVars.solarSystemId,
|
||||||
|
);
|
||||||
|
|
||||||
|
// console.log('JOipP', `render ${nodeVars.id}`, render++);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{nodeVars.visible && (
|
{nodeVars.visible && (
|
||||||
<div className={classes.Bookmarks}>
|
<div className={classes.Bookmarks}>
|
||||||
|
{nodeVars.isShattered && (
|
||||||
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.shattered, '!pr-[2px]')}>
|
||||||
|
<WdTooltipWrapper content="Shattered" position={TooltipPosition.top}>
|
||||||
|
<span className={clsx('block w-[10px] h-[10px]', classes.ShatteredIcon)} />
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{localKillsCount != null && localKillsCount > 0 && nodeVars.solarSystemId && localKillsActivityType && (
|
||||||
|
<KillsCounter
|
||||||
|
killsCount={localKillsCount}
|
||||||
|
systemId={nodeVars.solarSystemId}
|
||||||
|
size={TooltipSize.lg}
|
||||||
|
killsActivityType={localKillsActivityType}
|
||||||
|
className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[localKillsActivityType])}
|
||||||
|
>
|
||||||
|
<div className={clsx(classes.BookmarkWithIcon)}>
|
||||||
|
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
||||||
|
<span className={clsx(classes.text)}>{localKillsCount}</span>
|
||||||
|
</div>
|
||||||
|
</KillsCounter>
|
||||||
|
)}
|
||||||
|
|
||||||
{nodeVars.labelCustom !== '' && (
|
{nodeVars.labelCustom !== '' && (
|
||||||
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
||||||
<span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]">{nodeVars.labelCustom}</span>
|
<span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]">{nodeVars.labelCustom}</span>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{nodeVars.isShattered && (
|
|
||||||
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.shattered)}>
|
|
||||||
<span className={clsx('pi pi-chart-pie', classes.icon)} />
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{localKillsCount && localKillsCount > 0 && nodeVars.solarSystemId && (
|
|
||||||
<KillsCounter
|
|
||||||
killsCount={localKillsCount}
|
|
||||||
systemId={nodeVars.solarSystemId}
|
|
||||||
size="lg"
|
|
||||||
killsActivityType={nodeVars.killsActivityType}
|
|
||||||
className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[nodeVars.killsActivityType!])}
|
|
||||||
>
|
|
||||||
<div className={clsx(classes.BookmarkWithIcon)}>
|
|
||||||
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
|
||||||
<span className={clsx(classes.text)}>{nodeVars.killsCount}</span>
|
|
||||||
</div>
|
|
||||||
</KillsCounter>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{nodeVars.labelsInfo.map(x => (
|
{nodeVars.labelsInfo.map(x => (
|
||||||
<div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
|
<div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
|
||||||
{x.shortName}
|
{x.shortName}
|
||||||
@@ -63,8 +73,11 @@ export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>
|
|||||||
className={clsx(
|
className={clsx(
|
||||||
classes.RootCustomNode,
|
classes.RootCustomNode,
|
||||||
nodeVars.regionClass && classes[nodeVars.regionClass],
|
nodeVars.regionClass && classes[nodeVars.regionClass],
|
||||||
nodeVars.status !== undefined ? classes[STATUS_CLASSES[nodeVars.status]] : '',
|
nodeVars.status !== undefined && classes[STATUS_CLASSES[nodeVars.status]],
|
||||||
{ [classes.selected]: nodeVars.selected },
|
{
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.rally]: nodeVars.isRally,
|
||||||
|
},
|
||||||
)}
|
)}
|
||||||
onMouseDownCapture={e => nodeVars.dbClick(e)}
|
onMouseDownCapture={e => nodeVars.dbClick(e)}
|
||||||
>
|
>
|
||||||
@@ -82,7 +95,11 @@ export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{nodeVars.tag != null && nodeVars.tag !== '' && (
|
{nodeVars.tag != null && nodeVars.tag !== '' && (
|
||||||
<div className={clsx(classes.TagTitle, 'text-sky-400 font-medium')}>{nodeVars.tag}</div>
|
<Tag
|
||||||
|
value={nodeVars.tag}
|
||||||
|
severity="warning"
|
||||||
|
className="py-0 px-[2px] text-[9px] [&_.p-tag-value]:leading-[1.3]"
|
||||||
|
></Tag>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<div
|
<div
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
@import './SolarSystemNodeDefault.module.scss';
|
@use './SolarSystemNodeDefault.module.scss';
|
||||||
|
|
||||||
/* ---------------------------------------------
|
/* ---------------------------------------------
|
||||||
Only override what's different from the base
|
Only override what's different from the base
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import { Handle, NodeProps, Position } from 'reactflow';
|
|||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import classes from './SolarSystemNodeTheme.module.scss';
|
import classes from './SolarSystemNodeTheme.module.scss';
|
||||||
import { PrimeIcons } from 'primereact/api';
|
import { PrimeIcons } from 'primereact/api';
|
||||||
import { useLocalCounter, useNodeKillsCount, useSolarSystemNode } from '../../hooks/useSolarSystemLogic';
|
import { useLocalCounter, useNodeKillsCount, useSolarSystemNode } from '../../hooks';
|
||||||
import {
|
import {
|
||||||
EFFECT_BACKGROUND_STYLES,
|
EFFECT_BACKGROUND_STYLES,
|
||||||
MARKER_BOOKMARK_BG_STYLES,
|
MARKER_BOOKMARK_BG_STYLES,
|
||||||
@@ -12,45 +12,54 @@ import {
|
|||||||
} from '@/hooks/Mapper/components/map/constants';
|
} from '@/hooks/Mapper/components/map/constants';
|
||||||
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
||||||
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
||||||
import { LocalCounter } from './SolarSystemLocalCounter';
|
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
||||||
import { KillsCounter } from './SolarSystemKillsCounter';
|
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
|
||||||
|
import { LocalCounter } from '@/hooks/Mapper/components/map/components/LocalCounter';
|
||||||
|
import { KillsCounter } from '@/hooks/Mapper/components/map/components/KillsCounter';
|
||||||
|
|
||||||
|
// let render = 0;
|
||||||
export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>) => {
|
export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>) => {
|
||||||
const nodeVars = useSolarSystemNode(props);
|
const nodeVars = useSolarSystemNode(props);
|
||||||
const { localCounterCharacters } = useLocalCounter(nodeVars);
|
const { localCounterCharacters } = useLocalCounter(nodeVars);
|
||||||
const localKillsCount = useNodeKillsCount(nodeVars.solarSystemId, nodeVars.killsCount);
|
const { killsCount: localKillsCount, killsActivityType: localKillsActivityType } = useNodeKillsCount(
|
||||||
|
nodeVars.solarSystemId,
|
||||||
|
);
|
||||||
|
|
||||||
|
// console.log('JOipP', `render ${nodeVars.id}`, render++);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{nodeVars.visible && (
|
{nodeVars.visible && (
|
||||||
<div className={classes.Bookmarks}>
|
<div className={classes.Bookmarks}>
|
||||||
|
{nodeVars.isShattered && (
|
||||||
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.shattered, '!pr-[2px]')}>
|
||||||
|
<WdTooltipWrapper content="Shattered" position={TooltipPosition.top}>
|
||||||
|
<span className={clsx('block w-[10px] h-[10px]', classes.ShatteredIcon)} />
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{localKillsCount && localKillsCount > 0 && nodeVars.solarSystemId && localKillsActivityType && (
|
||||||
|
<KillsCounter
|
||||||
|
killsCount={localKillsCount}
|
||||||
|
systemId={nodeVars.solarSystemId}
|
||||||
|
size={TooltipSize.lg}
|
||||||
|
killsActivityType={localKillsActivityType}
|
||||||
|
className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[localKillsActivityType])}
|
||||||
|
>
|
||||||
|
<div className={clsx(classes.BookmarkWithIcon)}>
|
||||||
|
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
||||||
|
<span className={clsx(classes.text)}>{localKillsCount}</span>
|
||||||
|
</div>
|
||||||
|
</KillsCounter>
|
||||||
|
)}
|
||||||
|
|
||||||
{nodeVars.labelCustom !== '' && (
|
{nodeVars.labelCustom !== '' && (
|
||||||
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
||||||
<span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]">{nodeVars.labelCustom}</span>
|
<span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]">{nodeVars.labelCustom}</span>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{nodeVars.isShattered && (
|
|
||||||
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.shattered)}>
|
|
||||||
<span className={clsx('pi pi-chart-pie', classes.icon)} />
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{localKillsCount && localKillsCount > 0 && nodeVars.solarSystemId && (
|
|
||||||
<KillsCounter
|
|
||||||
killsCount={localKillsCount}
|
|
||||||
systemId={nodeVars.solarSystemId}
|
|
||||||
size="lg"
|
|
||||||
killsActivityType={nodeVars.killsActivityType}
|
|
||||||
className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[nodeVars.killsActivityType!])}
|
|
||||||
>
|
|
||||||
<div className={clsx(classes.BookmarkWithIcon)}>
|
|
||||||
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
|
||||||
<span className={clsx(classes.text)}>{nodeVars.killsCount}</span>
|
|
||||||
</div>
|
|
||||||
</KillsCounter>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{nodeVars.labelsInfo.map(x => (
|
{nodeVars.labelsInfo.map(x => (
|
||||||
<div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
|
<div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
|
||||||
{x.shortName}
|
{x.shortName}
|
||||||
@@ -64,7 +73,10 @@ export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>)
|
|||||||
classes.RootCustomNode,
|
classes.RootCustomNode,
|
||||||
nodeVars.regionClass && classes[nodeVars.regionClass],
|
nodeVars.regionClass && classes[nodeVars.regionClass],
|
||||||
nodeVars.status !== undefined ? classes[STATUS_CLASSES[nodeVars.status]] : '',
|
nodeVars.status !== undefined ? classes[STATUS_CLASSES[nodeVars.status]] : '',
|
||||||
{ [classes.selected]: nodeVars.selected },
|
{
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.rally]: nodeVars.isRally,
|
||||||
|
},
|
||||||
)}
|
)}
|
||||||
onMouseDownCapture={e => nodeVars.dbClick(e)}
|
onMouseDownCapture={e => nodeVars.dbClick(e)}
|
||||||
>
|
>
|
||||||
@@ -109,23 +121,13 @@ export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>)
|
|||||||
|
|
||||||
<div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
|
<div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
|
||||||
{nodeVars.customName && (
|
{nodeVars.customName && (
|
||||||
<div
|
<div className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
|
||||||
className={clsx(
|
|
||||||
classes.CustomName,
|
|
||||||
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] whitespace-nowrap overflow-hidden text-ellipsis mr-0.5',
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{nodeVars.customName}
|
{nodeVars.customName}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{!nodeVars.isWormhole && !nodeVars.customName && (
|
{!nodeVars.isWormhole && !nodeVars.customName && (
|
||||||
<div
|
<div className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
|
||||||
className={clsx(
|
|
||||||
classes.RegionName,
|
|
||||||
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] whitespace-nowrap overflow-hidden text-ellipsis mr-0.5',
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{nodeVars.regionName}
|
{nodeVars.regionName}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
@use '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
||||||
|
|
||||||
.Signature {
|
.Signature {
|
||||||
position: relative;
|
position: relative;
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ export enum SOLAR_SYSTEM_CLASS_IDS {
|
|||||||
thera = 12,
|
thera = 12,
|
||||||
c13 = 13,
|
c13 = 13,
|
||||||
sentinel = 14,
|
sentinel = 14,
|
||||||
baribican = 15,
|
barbican = 15,
|
||||||
vidette = 16,
|
vidette = 16,
|
||||||
conflux = 17,
|
conflux = 17,
|
||||||
redoubt = 18,
|
redoubt = 18,
|
||||||
@@ -82,7 +82,7 @@ export const SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS = {
|
|||||||
thera: SOLAR_SYSTEM_CLASS_GROUPS.thera,
|
thera: SOLAR_SYSTEM_CLASS_GROUPS.thera,
|
||||||
c13: SOLAR_SYSTEM_CLASS_GROUPS.c13,
|
c13: SOLAR_SYSTEM_CLASS_GROUPS.c13,
|
||||||
sentinel: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
sentinel: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
baribican: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
barbican: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
vidette: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
vidette: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
conflux: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
conflux: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
redoubt: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
redoubt: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
@@ -217,7 +217,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 14,
|
wormholeClassID: 14,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 14 (Sentinel Drifter)',
|
title: 'Class 14 (Sentinel Drifter)',
|
||||||
shortTitle: 'Sentinel',
|
shortTitle: 'Sentinel MZ',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'barbican',
|
id: 'barbican',
|
||||||
@@ -225,7 +225,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 15,
|
wormholeClassID: 15,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 15 (Barbican Drifter)',
|
title: 'Class 15 (Barbican Drifter)',
|
||||||
shortTitle: 'Barbican',
|
shortTitle: 'Liberated Barbican',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'vidette',
|
id: 'vidette',
|
||||||
@@ -233,7 +233,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 16,
|
wormholeClassID: 16,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 16 (Vidette Drifter)',
|
title: 'Class 16 (Vidette Drifter)',
|
||||||
shortTitle: 'Vidette',
|
shortTitle: 'Sanctified Vidette',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'conflux',
|
id: 'conflux',
|
||||||
@@ -241,7 +241,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 17,
|
wormholeClassID: 17,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 17 (Conflux Drifter)',
|
title: 'Class 17 (Conflux Drifter)',
|
||||||
shortTitle: 'Conflux',
|
shortTitle: 'Conflux Eyrie',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'redoubt',
|
id: 'redoubt',
|
||||||
@@ -249,7 +249,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 18,
|
wormholeClassID: 18,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 18 (Redoubt Drifter)',
|
title: 'Class 18 (Redoubt Drifter)',
|
||||||
shortTitle: 'Redoubt',
|
shortTitle: 'Azdaja Redoubt',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'a1',
|
id: 'a1',
|
||||||
@@ -322,6 +322,9 @@ export const WORMHOLES_ADDITIONAL_INFO: Record<string, WormholesAdditionalInfoTy
|
|||||||
export const WORMHOLES_ADDITIONAL_INFO_BY_CLASS_ID: Record<string, WormholesAdditionalInfoType> =
|
export const WORMHOLES_ADDITIONAL_INFO_BY_CLASS_ID: Record<string, WormholesAdditionalInfoType> =
|
||||||
WORMHOLES_ADDITIONAL_INFO_RAW.reduce((acc, x) => ({ ...acc, [x.wormholeClassID]: x }), {});
|
WORMHOLES_ADDITIONAL_INFO_RAW.reduce((acc, x) => ({ ...acc, [x.wormholeClassID]: x }), {});
|
||||||
|
|
||||||
|
export const WORMHOLES_ADDITIONAL_INFO_BY_SHORT_NAME: Record<string, WormholesAdditionalInfoType> =
|
||||||
|
WORMHOLES_ADDITIONAL_INFO_RAW.reduce((acc, x) => ({ ...acc, [x.shortName]: x }), {});
|
||||||
|
|
||||||
// export const SOLAR_SYSTEM_CLASS_NAMES = {
|
// export const SOLAR_SYSTEM_CLASS_NAMES = {
|
||||||
// ccp1 = ,
|
// ccp1 = ,
|
||||||
// c1 = ,
|
// c1 = ,
|
||||||
@@ -650,6 +653,7 @@ export enum LABELS {
|
|||||||
l3 = '3',
|
l3 = '3',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
export const LABELS_INFO: Record<string, any> = {
|
export const LABELS_INFO: Record<string, any> = {
|
||||||
[LABELS.clear]: { id: 'clear', name: 'Clear', shortName: '', icon: '' },
|
[LABELS.clear]: { id: 'clear', name: 'Clear', shortName: '', icon: '' },
|
||||||
[LABELS.la]: { id: 'la', name: 'Label A', shortName: 'A', icon: '' },
|
[LABELS.la]: { id: 'la', name: 'Label A', shortName: 'A', icon: '' },
|
||||||
@@ -750,6 +754,17 @@ export const SHIP_SIZES_SIZE = {
|
|||||||
[ShipSizeStatus.capital]: '2M',
|
[ShipSizeStatus.capital]: '2M',
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const SHIP_MASSES_SIZE: Record<number, ShipSizeStatus> = {
|
||||||
|
5_000_000: ShipSizeStatus.small,
|
||||||
|
62_000_000: ShipSizeStatus.medium,
|
||||||
|
300_000_000: ShipSizeStatus.large,
|
||||||
|
375_000_000: ShipSizeStatus.large,
|
||||||
|
1_000_000_000: ShipSizeStatus.freight,
|
||||||
|
1_350_000_000: ShipSizeStatus.capital,
|
||||||
|
1_800_000_000: ShipSizeStatus.capital,
|
||||||
|
2_000_000_000: ShipSizeStatus.capital,
|
||||||
|
};
|
||||||
|
|
||||||
export const SHIP_SIZES_DESCRIPTION = {
|
export const SHIP_SIZES_DESCRIPTION = {
|
||||||
[ShipSizeStatus.small]: 'Frigate wormhole - up to Destroyer | 5K t.',
|
[ShipSizeStatus.small]: 'Frigate wormhole - up to Destroyer | 5K t.',
|
||||||
[ShipSizeStatus.medium]: 'Cruise wormhole - up to Battlecruiser | 62K t.',
|
[ShipSizeStatus.medium]: 'Cruise wormhole - up to Battlecruiser | 62K t.',
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ export const isWormholeSpace = (wormholeClassID: number) => {
|
|||||||
case SOLAR_SYSTEM_CLASS_IDS.c6:
|
case SOLAR_SYSTEM_CLASS_IDS.c6:
|
||||||
case SOLAR_SYSTEM_CLASS_IDS.c13:
|
case SOLAR_SYSTEM_CLASS_IDS.c13:
|
||||||
case SOLAR_SYSTEM_CLASS_IDS.thera:
|
case SOLAR_SYSTEM_CLASS_IDS.thera:
|
||||||
case SOLAR_SYSTEM_CLASS_IDS.baribican:
|
case SOLAR_SYSTEM_CLASS_IDS.barbican:
|
||||||
case SOLAR_SYSTEM_CLASS_IDS.vidette:
|
case SOLAR_SYSTEM_CLASS_IDS.vidette:
|
||||||
case SOLAR_SYSTEM_CLASS_IDS.conflux:
|
case SOLAR_SYSTEM_CLASS_IDS.conflux:
|
||||||
case SOLAR_SYSTEM_CLASS_IDS.redoubt:
|
case SOLAR_SYSTEM_CLASS_IDS.redoubt:
|
||||||
|
|||||||
@@ -6,5 +6,5 @@ export * from './useCommandsCharacters';
|
|||||||
export * from './useCommandsConnections';
|
export * from './useCommandsConnections';
|
||||||
export * from './useCommandsConnections';
|
export * from './useCommandsConnections';
|
||||||
export * from './useCenterSystem';
|
export * from './useCenterSystem';
|
||||||
export * from './useSelectSystem';
|
export * from './useSelectSystems';
|
||||||
export * from './useMapCommands';
|
export * from './useMapCommands';
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
||||||
import { useCallback, useRef } from 'react';
|
|
||||||
import {
|
import {
|
||||||
CommandCharacterAdded,
|
CommandCharacterAdded,
|
||||||
CommandCharacterRemoved,
|
CommandCharacterRemoved,
|
||||||
@@ -7,6 +6,7 @@ import {
|
|||||||
CommandCharacterUpdated,
|
CommandCharacterUpdated,
|
||||||
CommandPresentCharacters,
|
CommandPresentCharacters,
|
||||||
} from '@/hooks/Mapper/types';
|
} from '@/hooks/Mapper/types';
|
||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
|
||||||
export const useCommandsCharacters = () => {
|
export const useCommandsCharacters = () => {
|
||||||
const { update } = useMapState();
|
const { update } = useMapState();
|
||||||
|
|||||||
@@ -2,10 +2,13 @@ import { Node, useReactFlow } from 'reactflow';
|
|||||||
import { useCallback, useRef } from 'react';
|
import { useCallback, useRef } from 'react';
|
||||||
import { CommandAddSystems } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { CommandAddSystems } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { convertSystem2Node } from '../../helpers';
|
import { convertSystem2Node } from '../../helpers';
|
||||||
|
import { useLoadSystemStatic } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
|
||||||
|
|
||||||
export const useMapAddSystems = () => {
|
export const useMapAddSystems = () => {
|
||||||
const rf = useReactFlow();
|
const rf = useReactFlow();
|
||||||
|
|
||||||
|
const { addSystemStatic } = useLoadSystemStatic({ systems: [] });
|
||||||
|
|
||||||
const ref = useRef({ rf });
|
const ref = useRef({ rf });
|
||||||
ref.current = { rf };
|
ref.current = { rf };
|
||||||
|
|
||||||
@@ -13,7 +16,10 @@ export const useMapAddSystems = () => {
|
|||||||
const { rf } = ref.current;
|
const { rf } = ref.current;
|
||||||
const nodes = rf.getNodes();
|
const nodes = rf.getNodes();
|
||||||
|
|
||||||
const prepared: Node[] = systems.filter(x => !nodes.some(y => x.id === y.id)).map(convertSystem2Node);
|
const newSystems = systems.filter(x => !nodes.some(y => x.id === y.id));
|
||||||
|
newSystems.forEach(x => addSystemStatic(x.system_static_info));
|
||||||
|
|
||||||
|
const prepared: Node[] = newSystems.map(convertSystem2Node);
|
||||||
rf.addNodes(prepared);
|
rf.addNodes(prepared);
|
||||||
}, []);
|
}, []);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { MapData, useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
import { MapData, useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
||||||
import { useCallback, useRef } from 'react';
|
|
||||||
import { CommandKillsUpdated, CommandMapUpdated } from '@/hooks/Mapper/types';
|
import { CommandKillsUpdated, CommandMapUpdated } from '@/hooks/Mapper/types';
|
||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
|
||||||
export const useMapCommands = () => {
|
export const useMapCommands = () => {
|
||||||
const { update } = useMapState();
|
const { update } = useMapState();
|
||||||
@@ -8,13 +8,21 @@ export const useMapCommands = () => {
|
|||||||
const ref = useRef({ update });
|
const ref = useRef({ update });
|
||||||
ref.current = { update };
|
ref.current = { update };
|
||||||
|
|
||||||
const mapUpdated = useCallback(({ hubs }: CommandMapUpdated) => {
|
const mapUpdated = useCallback(({ hubs, system_signatures, kills }: CommandMapUpdated) => {
|
||||||
const out: Partial<MapData> = {};
|
const out: Partial<MapData> = {};
|
||||||
|
|
||||||
if (hubs) {
|
if (hubs) {
|
||||||
out.hubs = hubs;
|
out.hubs = hubs;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (system_signatures) {
|
||||||
|
out.systemSignatures = system_signatures;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (kills) {
|
||||||
|
out.kills = kills.reduce((acc, x) => ({ ...acc, [x.solar_system_id]: x.kills }), {});
|
||||||
|
}
|
||||||
|
|
||||||
ref.current.update(out);
|
ref.current.update(out);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,10 @@
|
|||||||
import { useReactFlow } from 'reactflow';
|
|
||||||
import { useCallback, useRef } from 'react';
|
|
||||||
import { CommandInit } from '@/hooks/Mapper/types/mapHandlers.ts';
|
|
||||||
import { convertConnection2Edge, convertSystem2Node } from '../../helpers';
|
|
||||||
import { MapData, useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
import { MapData, useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
||||||
|
import { useEventBuffer } from '@/hooks/Mapper/hooks';
|
||||||
|
import { SolarSystemConnection, SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
|
import { CommandInit } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
import { useReactFlow } from 'reactflow';
|
||||||
|
import { convertConnection2Edge, convertSystem2Node } from '../../helpers';
|
||||||
|
|
||||||
export const useMapInit = () => {
|
export const useMapInit = () => {
|
||||||
const rf = useReactFlow();
|
const rf = useReactFlow();
|
||||||
@@ -11,9 +13,24 @@ export const useMapInit = () => {
|
|||||||
const ref = useRef({ rf, data, update });
|
const ref = useRef({ rf, data, update });
|
||||||
ref.current = { update, data, rf };
|
ref.current = { update, data, rf };
|
||||||
|
|
||||||
|
const updateSystems = useCallback((systems: SolarSystemRawType[]) => {
|
||||||
|
const { rf } = ref.current;
|
||||||
|
rf.setNodes(systems.map(convertSystem2Node));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const { handleEvent: handleUpdateSystems } = useEventBuffer<any>(updateSystems);
|
||||||
|
|
||||||
|
const updateEdges = useCallback((connections: SolarSystemConnection[]) => {
|
||||||
|
const { rf } = ref.current;
|
||||||
|
rf.setEdges(connections.map(convertConnection2Edge));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const { handleEvent: handleUpdateConnections } = useEventBuffer<any>(updateEdges);
|
||||||
|
|
||||||
return useCallback(
|
return useCallback(
|
||||||
({
|
({
|
||||||
systems,
|
systems,
|
||||||
|
system_signatures,
|
||||||
kills,
|
kills,
|
||||||
connections,
|
connections,
|
||||||
wormholes,
|
wormholes,
|
||||||
@@ -23,7 +40,6 @@ export const useMapInit = () => {
|
|||||||
hubs,
|
hubs,
|
||||||
}: CommandInit) => {
|
}: CommandInit) => {
|
||||||
const { update } = ref.current;
|
const { update } = ref.current;
|
||||||
const { rf } = ref.current;
|
|
||||||
|
|
||||||
const updateData: Partial<MapData> = {};
|
const updateData: Partial<MapData> = {};
|
||||||
|
|
||||||
@@ -51,6 +67,10 @@ export const useMapInit = () => {
|
|||||||
updateData.systems = systems;
|
updateData.systems = systems;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (system_signatures) {
|
||||||
|
updateData.systemSignatures = system_signatures;
|
||||||
|
}
|
||||||
|
|
||||||
if (kills) {
|
if (kills) {
|
||||||
updateData.kills = kills.reduce((acc, x) => ({ ...acc, [x.solar_system_id]: x.kills }), {});
|
updateData.kills = kills.reduce((acc, x) => ({ ...acc, [x.solar_system_id]: x.kills }), {});
|
||||||
}
|
}
|
||||||
@@ -58,11 +78,13 @@ export const useMapInit = () => {
|
|||||||
update(updateData);
|
update(updateData);
|
||||||
|
|
||||||
if (systems) {
|
if (systems) {
|
||||||
rf.setNodes(systems.map(convertSystem2Node));
|
handleUpdateSystems(systems);
|
||||||
|
// rf.setNodes(systems.map(convertSystem2Node));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (connections) {
|
if (connections) {
|
||||||
rf.setEdges(connections.map(convertConnection2Edge));
|
handleUpdateConnections(connections);
|
||||||
|
// rf.setEdges(connections.map(convertConnection2Edge));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
[],
|
[],
|
||||||
|
|||||||
@@ -1,21 +0,0 @@
|
|||||||
import { useReactFlow } from 'reactflow';
|
|
||||||
import { useCallback, useRef } from 'react';
|
|
||||||
import { CommandSelectSystem } from '@/hooks/Mapper/types';
|
|
||||||
|
|
||||||
export const useSelectSystem = () => {
|
|
||||||
const rf = useReactFlow();
|
|
||||||
|
|
||||||
const ref = useRef({ rf });
|
|
||||||
ref.current = { rf };
|
|
||||||
|
|
||||||
return useCallback((systemId: CommandSelectSystem) => {
|
|
||||||
ref.current.rf.setNodes(nds =>
|
|
||||||
nds.map(node => {
|
|
||||||
return {
|
|
||||||
...node,
|
|
||||||
selected: node.id === systemId,
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}, []);
|
|
||||||
};
|
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
import { OnMapSelectionChange } from '@/hooks/Mapper/components/map/map.types.ts';
|
||||||
|
import { CommandSelectSystems } from '@/hooks/Mapper/types';
|
||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
import { useReactFlow } from 'reactflow';
|
||||||
|
|
||||||
|
export const useSelectSystems = (onSelectionChange: OnMapSelectionChange) => {
|
||||||
|
const rf = useReactFlow();
|
||||||
|
|
||||||
|
const ref = useRef({ rf, onSelectionChange });
|
||||||
|
ref.current = { rf, onSelectionChange };
|
||||||
|
|
||||||
|
return useCallback(({ systems, delay }: CommandSelectSystems) => {
|
||||||
|
const run = () => {
|
||||||
|
ref.current.rf.setNodes(nds =>
|
||||||
|
nds.map(node => {
|
||||||
|
return {
|
||||||
|
...node,
|
||||||
|
selected: systems.includes(node.id),
|
||||||
|
};
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (delay == null || delay === 0) {
|
||||||
|
run();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setTimeout(run, delay);
|
||||||
|
}, []);
|
||||||
|
};
|
||||||
@@ -1,3 +1,11 @@
|
|||||||
export * from './useMapHandlers';
|
export * from './useMapHandlers';
|
||||||
export * from './useUpdateNodes';
|
export * from './useUpdateNodes';
|
||||||
export * from './useNodesEdgesState';
|
export * from './useNodesEdgesState';
|
||||||
|
export * from './useBackgroundVars';
|
||||||
|
export * from './useKillsCounter';
|
||||||
|
export * from './useSystemName';
|
||||||
|
export * from './useNodesEdgesState';
|
||||||
|
export * from './useSolarSystemNode';
|
||||||
|
export * from './useUnsplashedSignatures';
|
||||||
|
export * from './useUpdateNodes';
|
||||||
|
export * from './useNodeKillsCount';
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { useSystemKills } from '../../mapInterface/widgets/SystemKills/hooks/useSystemKills';
|
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useSystemKills } from '@/hooks/Mapper/components/mapInterface/widgets/WSystemKills/hooks/useSystemKills.ts';
|
||||||
|
|
||||||
interface UseKillsCounterProps {
|
interface UseKillsCounterProps {
|
||||||
realSystemId: string;
|
realSystemId: string;
|
||||||
@@ -22,18 +22,18 @@ export function useKillsCounter({ realSystemId }: UseKillsCounterProps) {
|
|||||||
systemId: realSystemId,
|
systemId: realSystemId,
|
||||||
outCommand,
|
outCommand,
|
||||||
showAllVisible: false,
|
showAllVisible: false,
|
||||||
|
sinceHours: 1,
|
||||||
});
|
});
|
||||||
|
|
||||||
const filteredKills = useMemo(() => {
|
const filteredKills = useMemo(() => {
|
||||||
if (!allKills || allKills.length === 0) return [];
|
if (!allKills || allKills.length === 0) return [];
|
||||||
|
|
||||||
return [...allKills]
|
// Sort kills by time, most recent first, but don't limit the number of kills
|
||||||
.sort((a, b) => {
|
return [...allKills].sort((a, b) => {
|
||||||
const aTime = a.kill_time ? new Date(a.kill_time).getTime() : 0;
|
const aTime = a.kill_time ? new Date(a.kill_time).getTime() : 0;
|
||||||
const bTime = b.kill_time ? new Date(b.kill_time).getTime() : 0;
|
const bTime = b.kill_time ? new Date(b.kill_time).getTime() : 0;
|
||||||
return bTime - aTime;
|
return bTime - aTime;
|
||||||
})
|
});
|
||||||
.slice(0, 10);
|
|
||||||
}, [allKills]);
|
}, [allKills]);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|||||||
31
assets/js/hooks/Mapper/components/map/hooks/useLabelsInfo.ts
Normal file
31
assets/js/hooks/Mapper/components/map/hooks/useLabelsInfo.ts
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
import { useMemo } from 'react';
|
||||||
|
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager';
|
||||||
|
import { LABELS_INFO, LABELS_ORDER } from '@/hooks/Mapper/components/map/constants';
|
||||||
|
interface UseLabelsInfoParams {
|
||||||
|
labels: string | null;
|
||||||
|
linkedSigPrefix: string | null;
|
||||||
|
isShowLinkedSigId: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type LabelInfo = {
|
||||||
|
id: string;
|
||||||
|
shortName: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
function sortedLabels(labels: string[]): LabelInfo[] {
|
||||||
|
if (!labels) return [];
|
||||||
|
return LABELS_ORDER.filter(x => labels.includes(x)).map(x => LABELS_INFO[x] as LabelInfo);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useLabelsInfo({ labels, linkedSigPrefix, isShowLinkedSigId }: UseLabelsInfoParams) {
|
||||||
|
const labelsManager = useMemo(() => new LabelsManager(labels ?? ''), [labels]);
|
||||||
|
const labelsInfo = useMemo(() => sortedLabels(labelsManager.list), [labelsManager]);
|
||||||
|
const labelCustom = useMemo(() => {
|
||||||
|
if (isShowLinkedSigId && linkedSigPrefix) {
|
||||||
|
return labelsManager.customLabel ? `${linkedSigPrefix}・${labelsManager.customLabel}` : linkedSigPrefix;
|
||||||
|
}
|
||||||
|
return labelsManager.customLabel;
|
||||||
|
}, [linkedSigPrefix, isShowLinkedSigId, labelsManager]);
|
||||||
|
|
||||||
|
return { labelsInfo, labelCustom };
|
||||||
|
}
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
import { ForwardedRef, useImperativeHandle, useRef } from 'react';
|
|
||||||
import {
|
import {
|
||||||
CommandAddConnections,
|
CommandAddConnections,
|
||||||
CommandAddSystems,
|
CommandAddSystems,
|
||||||
@@ -14,12 +13,16 @@ import {
|
|||||||
CommandRemoveSystems,
|
CommandRemoveSystems,
|
||||||
Commands,
|
Commands,
|
||||||
CommandSelectSystem,
|
CommandSelectSystem,
|
||||||
|
CommandSelectSystems,
|
||||||
CommandUpdateConnection,
|
CommandUpdateConnection,
|
||||||
CommandUpdateSystems,
|
CommandUpdateSystems,
|
||||||
MapHandlers,
|
MapHandlers,
|
||||||
} from '@/hooks/Mapper/types/mapHandlers.ts';
|
} from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
import { ForwardedRef, useImperativeHandle, useRef } from 'react';
|
||||||
|
|
||||||
|
import { OnMapSelectionChange } from '@/hooks/Mapper/components/map/map.types.ts';
|
||||||
import {
|
import {
|
||||||
|
useCenterSystem,
|
||||||
useCommandsCharacters,
|
useCommandsCharacters,
|
||||||
useCommandsConnections,
|
useCommandsConnections,
|
||||||
useMapAddSystems,
|
useMapAddSystems,
|
||||||
@@ -27,10 +30,8 @@ import {
|
|||||||
useMapInit,
|
useMapInit,
|
||||||
useMapRemoveSystems,
|
useMapRemoveSystems,
|
||||||
useMapUpdateSystems,
|
useMapUpdateSystems,
|
||||||
useCenterSystem,
|
useSelectSystems,
|
||||||
useSelectSystem,
|
|
||||||
} from './api';
|
} from './api';
|
||||||
import { OnMapSelectionChange } from '@/hooks/Mapper/components/map/map.types.ts';
|
|
||||||
|
|
||||||
export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange: OnMapSelectionChange) => {
|
export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange: OnMapSelectionChange) => {
|
||||||
const mapInit = useMapInit();
|
const mapInit = useMapInit();
|
||||||
@@ -38,7 +39,7 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
|
|||||||
const mapUpdateSystems = useMapUpdateSystems();
|
const mapUpdateSystems = useMapUpdateSystems();
|
||||||
const removeSystems = useMapRemoveSystems(onSelectionChange);
|
const removeSystems = useMapRemoveSystems(onSelectionChange);
|
||||||
const centerSystem = useCenterSystem();
|
const centerSystem = useCenterSystem();
|
||||||
const selectSystem = useSelectSystem();
|
const selectSystems = useSelectSystems(onSelectionChange);
|
||||||
|
|
||||||
const selectRef = useRef({ onSelectionChange });
|
const selectRef = useRef({ onSelectionChange });
|
||||||
selectRef.current = { onSelectionChange };
|
selectRef.current = { onSelectionChange };
|
||||||
@@ -48,96 +49,87 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
|
|||||||
const { charactersUpdated, presentCharacters, characterAdded, characterRemoved, characterUpdated } =
|
const { charactersUpdated, presentCharacters, characterAdded, characterRemoved, characterUpdated } =
|
||||||
useCommandsCharacters();
|
useCommandsCharacters();
|
||||||
|
|
||||||
useImperativeHandle(
|
useImperativeHandle(ref, () => {
|
||||||
ref,
|
return {
|
||||||
() => {
|
command(type, data) {
|
||||||
return {
|
switch (type) {
|
||||||
command(type, data) {
|
case Commands.init:
|
||||||
switch (type) {
|
mapInit(data as CommandInit);
|
||||||
case Commands.init:
|
break;
|
||||||
mapInit(data as CommandInit);
|
case Commands.addSystems:
|
||||||
break;
|
setTimeout(() => mapAddSystems(data as CommandAddSystems), 100);
|
||||||
case Commands.addSystems:
|
break;
|
||||||
setTimeout(() => mapAddSystems(data as CommandAddSystems), 100);
|
case Commands.updateSystems:
|
||||||
break;
|
mapUpdateSystems(data as CommandUpdateSystems);
|
||||||
case Commands.updateSystems:
|
break;
|
||||||
mapUpdateSystems(data as CommandUpdateSystems);
|
case Commands.removeSystems:
|
||||||
break;
|
setTimeout(() => removeSystems(data as CommandRemoveSystems), 100);
|
||||||
case Commands.removeSystems:
|
break;
|
||||||
setTimeout(() => removeSystems(data as CommandRemoveSystems), 100);
|
case Commands.addConnections:
|
||||||
break;
|
setTimeout(() => addConnections(data as CommandAddConnections), 100);
|
||||||
case Commands.addConnections:
|
break;
|
||||||
setTimeout(() => addConnections(data as CommandAddConnections), 100);
|
case Commands.removeConnections:
|
||||||
break;
|
setTimeout(() => removeConnections(data as CommandRemoveConnections), 100);
|
||||||
case Commands.removeConnections:
|
break;
|
||||||
setTimeout(() => removeConnections(data as CommandRemoveConnections), 100);
|
case Commands.charactersUpdated:
|
||||||
break;
|
charactersUpdated(data as CommandCharactersUpdated);
|
||||||
case Commands.charactersUpdated:
|
break;
|
||||||
charactersUpdated(data as CommandCharactersUpdated);
|
case Commands.characterAdded:
|
||||||
break;
|
characterAdded(data as CommandCharacterAdded);
|
||||||
case Commands.characterAdded:
|
break;
|
||||||
characterAdded(data as CommandCharacterAdded);
|
case Commands.characterRemoved:
|
||||||
break;
|
characterRemoved(data as CommandCharacterRemoved);
|
||||||
case Commands.characterRemoved:
|
break;
|
||||||
characterRemoved(data as CommandCharacterRemoved);
|
case Commands.characterUpdated:
|
||||||
break;
|
characterUpdated(data as CommandCharacterUpdated);
|
||||||
case Commands.characterUpdated:
|
break;
|
||||||
characterUpdated(data as CommandCharacterUpdated);
|
case Commands.presentCharacters:
|
||||||
break;
|
presentCharacters(data as CommandPresentCharacters);
|
||||||
case Commands.presentCharacters:
|
break;
|
||||||
presentCharacters(data as CommandPresentCharacters);
|
case Commands.updateConnection:
|
||||||
break;
|
updateConnection(data as CommandUpdateConnection);
|
||||||
case Commands.updateConnection:
|
break;
|
||||||
updateConnection(data as CommandUpdateConnection);
|
case Commands.mapUpdated:
|
||||||
break;
|
mapUpdated(data as CommandMapUpdated);
|
||||||
case Commands.mapUpdated:
|
break;
|
||||||
mapUpdated(data as CommandMapUpdated);
|
case Commands.killsUpdated:
|
||||||
break;
|
killsUpdated(data as CommandKillsUpdated);
|
||||||
case Commands.killsUpdated:
|
break;
|
||||||
killsUpdated(data as CommandKillsUpdated);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case Commands.centerSystem:
|
case Commands.centerSystem:
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
const systemId = `${data}`;
|
const systemId = `${data}`;
|
||||||
centerSystem(systemId as CommandSelectSystem);
|
centerSystem(systemId as CommandSelectSystem);
|
||||||
}, 100);
|
}, 100);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Commands.selectSystem:
|
case Commands.selectSystem:
|
||||||
setTimeout(() => {
|
selectSystems({ systems: [data as string], delay: 500 });
|
||||||
const systemId = `${data}`;
|
break;
|
||||||
selectRef.current.onSelectionChange({
|
|
||||||
systems: [systemId],
|
|
||||||
connections: [],
|
|
||||||
});
|
|
||||||
selectSystem(systemId as CommandSelectSystem);
|
|
||||||
}, 500);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case Commands.routes:
|
case Commands.selectSystems:
|
||||||
// do nothing here
|
selectSystems(data as CommandSelectSystems);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Commands.signaturesUpdated:
|
case Commands.pingAdded:
|
||||||
// do nothing here
|
case Commands.pingCancelled:
|
||||||
break;
|
case Commands.routes:
|
||||||
|
case Commands.signaturesUpdated:
|
||||||
|
case Commands.linkSignatureToSystem:
|
||||||
|
case Commands.detailedKillsUpdated:
|
||||||
|
case Commands.characterActivityData:
|
||||||
|
case Commands.trackingCharactersData:
|
||||||
|
case Commands.updateActivity:
|
||||||
|
case Commands.updateTracking:
|
||||||
|
case Commands.userSettingsUpdated:
|
||||||
|
// do nothing
|
||||||
|
break;
|
||||||
|
|
||||||
case Commands.linkSignatureToSystem:
|
default:
|
||||||
// do nothing here
|
console.warn(`Map handlers: Unknown command: ${type}`, data);
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
case Commands.detailedKillsUpdated:
|
},
|
||||||
// do nothing here
|
};
|
||||||
break;
|
}, []);
|
||||||
|
|
||||||
default:
|
|
||||||
console.warn(`Map handlers: Unknown command: ${type}`, data);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
};
|
|
||||||
},
|
|
||||||
[],
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -0,0 +1,87 @@
|
|||||||
|
import { useEffect, useState, useCallback, useMemo } from 'react';
|
||||||
|
import { useMapEventListener } from '@/hooks/Mapper/events';
|
||||||
|
import { Commands } from '@/hooks/Mapper/types';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
|
interface Kill {
|
||||||
|
solar_system_id: number | string;
|
||||||
|
kills: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface MapEvent {
|
||||||
|
name: Commands;
|
||||||
|
data?: unknown;
|
||||||
|
payload?: Kill[];
|
||||||
|
}
|
||||||
|
|
||||||
|
function getActivityType(count: number): string {
|
||||||
|
if (count <= 5) return 'activityNormal';
|
||||||
|
if (count <= 30) return 'activityWarn';
|
||||||
|
return 'activityDanger';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useNodeKillsCount(systemId: number | string, initialKillsCount: number | null = null): { killsCount: number | null; killsActivityType: string | null } {
|
||||||
|
const [killsCount, setKillsCount] = useState<number | null>(initialKillsCount);
|
||||||
|
const { data: mapData } = useMapRootState();
|
||||||
|
const { detailedKills = {} } = mapData;
|
||||||
|
|
||||||
|
// Calculate 1-hour kill count from detailed kills
|
||||||
|
const oneHourKillCount = useMemo(() => {
|
||||||
|
const systemKills = detailedKills[systemId] || [];
|
||||||
|
|
||||||
|
// If we have detailed kills data (even if empty), use it for counting
|
||||||
|
if (Object.prototype.hasOwnProperty.call(detailedKills, systemId)) {
|
||||||
|
const oneHourAgo = Date.now() - 60 * 60 * 1000; // 1 hour in milliseconds
|
||||||
|
const recentKills = systemKills.filter(kill => {
|
||||||
|
if (!kill.kill_time) return false;
|
||||||
|
const killTime = new Date(kill.kill_time).getTime();
|
||||||
|
if (isNaN(killTime)) return false;
|
||||||
|
return killTime >= oneHourAgo;
|
||||||
|
});
|
||||||
|
|
||||||
|
return recentKills.length; // Return 0 if no recent kills, not null
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return null only if we don't have detailed kills data for this system
|
||||||
|
return null;
|
||||||
|
}, [detailedKills, systemId]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
// Always prefer the calculated 1-hour count over initial count
|
||||||
|
// This ensures we properly expire old kills
|
||||||
|
if (oneHourKillCount !== null) {
|
||||||
|
setKillsCount(oneHourKillCount);
|
||||||
|
} else if (detailedKills[systemId] && detailedKills[systemId].length === 0) {
|
||||||
|
// If we have detailed kills data but it's empty, set to 0
|
||||||
|
setKillsCount(0);
|
||||||
|
} else {
|
||||||
|
// Only fall back to initial count if we have no detailed kills data at all
|
||||||
|
setKillsCount(initialKillsCount);
|
||||||
|
}
|
||||||
|
}, [oneHourKillCount, initialKillsCount, detailedKills, systemId]);
|
||||||
|
|
||||||
|
const handleEvent = useCallback(
|
||||||
|
(event: MapEvent): boolean => {
|
||||||
|
if (event.name === Commands.killsUpdated && Array.isArray(event.payload)) {
|
||||||
|
const killForSystem = event.payload.find(kill => kill.solar_system_id.toString() === systemId.toString());
|
||||||
|
if (killForSystem && typeof killForSystem.kills === 'number') {
|
||||||
|
// Only update if we don't have detailed kills data
|
||||||
|
if (!detailedKills[systemId] || detailedKills[systemId].length === 0) {
|
||||||
|
setKillsCount(killForSystem.kills);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
},
|
||||||
|
[systemId, detailedKills],
|
||||||
|
);
|
||||||
|
|
||||||
|
useMapEventListener(handleEvent);
|
||||||
|
|
||||||
|
const killsActivityType = useMemo(() => {
|
||||||
|
return killsCount !== null && killsCount > 0 ? getActivityType(killsCount) : null;
|
||||||
|
}, [killsCount]);
|
||||||
|
|
||||||
|
return { killsCount, killsActivityType };
|
||||||
|
}
|
||||||
@@ -1,248 +1,19 @@
|
|||||||
import { useEffect, useMemo, useState } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { MapSolarSystemType } from '../map.types';
|
import { MapSolarSystemType } from '../map.types';
|
||||||
import { NodeProps } from 'reactflow';
|
import { NodeProps } from 'reactflow';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
import { useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider';
|
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider';
|
||||||
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick';
|
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick';
|
||||||
import { REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
|
import { Regions, REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
|
||||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace';
|
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace';
|
||||||
import { getSystemClassStyles, prepareUnsplashedChunks } from '@/hooks/Mapper/components/map/helpers';
|
import { getSystemClassStyles } from '@/hooks/Mapper/components/map/helpers';
|
||||||
import { sortWHClasses } from '@/hooks/Mapper/helpers';
|
import { sortWHClasses } from '@/hooks/Mapper/helpers';
|
||||||
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager';
|
import { CharacterTypeRaw, OutCommand, PingType, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
import { CharacterTypeRaw, Commands, OutCommand, SystemSignature } from '@/hooks/Mapper/types';
|
import { useUnsplashedSignatures } from './useUnsplashedSignatures';
|
||||||
import { LABELS_INFO, LABELS_ORDER } from '@/hooks/Mapper/components/map/constants';
|
import { useSystemName } from './useSystemName';
|
||||||
import { useMapEventListener } from '@/hooks/Mapper/events';
|
import { LabelInfo, useLabelsInfo } from './useLabelsInfo';
|
||||||
|
import { getSystemStaticInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
|
||||||
export type LabelInfo = {
|
|
||||||
id: string;
|
|
||||||
shortName: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
export type UnsplashedSignatureType = SystemSignature & { sig_id: string };
|
|
||||||
|
|
||||||
function getActivityType(count: number): string {
|
|
||||||
if (count <= 5) return 'activityNormal';
|
|
||||||
if (count <= 30) return 'activityWarn';
|
|
||||||
return 'activityDanger';
|
|
||||||
}
|
|
||||||
|
|
||||||
const SpaceToClass: Record<string, string> = {
|
|
||||||
[Spaces.Caldari]: 'Caldaria',
|
|
||||||
[Spaces.Matar]: 'Mataria',
|
|
||||||
[Spaces.Amarr]: 'Amarria',
|
|
||||||
[Spaces.Gallente]: 'Gallente',
|
|
||||||
};
|
|
||||||
|
|
||||||
function sortedLabels(labels: string[]): LabelInfo[] {
|
|
||||||
if (!labels) return [];
|
|
||||||
return LABELS_ORDER.filter(x => labels.includes(x)).map(x => LABELS_INFO[x] as LabelInfo);
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useLocalCounter(nodeVars: SolarSystemNodeVars) {
|
|
||||||
const localCounterCharacters = useMemo(() => {
|
|
||||||
return nodeVars.charactersInSystem
|
|
||||||
.map(char => ({
|
|
||||||
...char,
|
|
||||||
compact: true,
|
|
||||||
isOwn: nodeVars.userCharacters.includes(char.eve_id),
|
|
||||||
}))
|
|
||||||
.sort((a, b) => a.name.localeCompare(b.name));
|
|
||||||
}, [nodeVars.charactersInSystem, nodeVars.userCharacters]);
|
|
||||||
return { localCounterCharacters };
|
|
||||||
}
|
|
||||||
|
|
||||||
export function useSolarSystemNode(props: NodeProps<MapSolarSystemType>): SolarSystemNodeVars {
|
|
||||||
const { id, data, selected } = props;
|
|
||||||
const {
|
|
||||||
system_static_info,
|
|
||||||
system_signatures,
|
|
||||||
locked,
|
|
||||||
name,
|
|
||||||
tag,
|
|
||||||
status,
|
|
||||||
labels,
|
|
||||||
temporary_name,
|
|
||||||
linked_sig_eve_id: linkedSigEveId = '',
|
|
||||||
} = data;
|
|
||||||
|
|
||||||
const {
|
|
||||||
system_class,
|
|
||||||
security,
|
|
||||||
class_title,
|
|
||||||
solar_system_id,
|
|
||||||
statics,
|
|
||||||
effect_name,
|
|
||||||
region_name,
|
|
||||||
region_id,
|
|
||||||
is_shattered,
|
|
||||||
solar_system_name,
|
|
||||||
} = system_static_info;
|
|
||||||
|
|
||||||
const {
|
|
||||||
interfaceSettings,
|
|
||||||
data: { systemSignatures: mapSystemSignatures },
|
|
||||||
} = useMapRootState();
|
|
||||||
|
|
||||||
const { isShowUnsplashedSignatures } = interfaceSettings;
|
|
||||||
const isTempSystemNameEnabled = useMapGetOption('show_temp_system_name') === 'true';
|
|
||||||
const isShowLinkedSigId = useMapGetOption('show_linked_signature_id') === 'true';
|
|
||||||
const isShowLinkedSigIdTempName = useMapGetOption('show_linked_signature_id_temp_name') === 'true';
|
|
||||||
|
|
||||||
const {
|
|
||||||
data: {
|
|
||||||
characters,
|
|
||||||
wormholesData,
|
|
||||||
hubs,
|
|
||||||
kills,
|
|
||||||
userCharacters,
|
|
||||||
isConnecting,
|
|
||||||
hoverNodeId,
|
|
||||||
visibleNodes,
|
|
||||||
showKSpaceBG,
|
|
||||||
isThickConnections,
|
|
||||||
},
|
|
||||||
outCommand,
|
|
||||||
} = useMapState();
|
|
||||||
|
|
||||||
const visible = useMemo(() => visibleNodes.has(id), [id, visibleNodes]);
|
|
||||||
|
|
||||||
const systemSigs = useMemo(
|
|
||||||
() => mapSystemSignatures[solar_system_id] || system_signatures,
|
|
||||||
[system_signatures, solar_system_id, mapSystemSignatures],
|
|
||||||
);
|
|
||||||
|
|
||||||
const charactersInSystem = useMemo(() => {
|
|
||||||
return characters.filter(c => c.location?.solar_system_id === solar_system_id && c.online);
|
|
||||||
}, [characters, solar_system_id]);
|
|
||||||
|
|
||||||
const isWormhole = isWormholeSpace(system_class);
|
|
||||||
|
|
||||||
const classTitleColor = useMemo(
|
|
||||||
() => getSystemClassStyles({ systemClass: system_class, security }),
|
|
||||||
[security, system_class],
|
|
||||||
);
|
|
||||||
|
|
||||||
const sortedStatics = useMemo(() => sortWHClasses(wormholesData, statics), [wormholesData, statics]);
|
|
||||||
|
|
||||||
const linkedSigPrefix = useMemo(() => (linkedSigEveId ? linkedSigEveId.split('-')[0] : null), [linkedSigEveId]);
|
|
||||||
|
|
||||||
const labelsManager = useMemo(() => new LabelsManager(labels ?? ''), [labels]);
|
|
||||||
const labelsInfo = useMemo(() => sortedLabels(labelsManager.list), [labelsManager]);
|
|
||||||
const labelCustom = useMemo(() => {
|
|
||||||
if (isShowLinkedSigId && linkedSigPrefix) {
|
|
||||||
return labelsManager.customLabel ? `${linkedSigPrefix}・${labelsManager.customLabel}` : linkedSigPrefix;
|
|
||||||
}
|
|
||||||
return labelsManager.customLabel;
|
|
||||||
}, [linkedSigPrefix, isShowLinkedSigId, labelsManager]);
|
|
||||||
|
|
||||||
const killsCount = useMemo(() => kills[solar_system_id] ?? null, [kills, solar_system_id]);
|
|
||||||
const killsActivityType = killsCount ? getActivityType(killsCount) : null;
|
|
||||||
|
|
||||||
const hasUserCharacters = useMemo(() => {
|
|
||||||
return charactersInSystem.some(x => userCharacters.includes(x.eve_id));
|
|
||||||
}, [charactersInSystem, userCharacters]);
|
|
||||||
|
|
||||||
const dbClick = useDoubleClick(() => {
|
|
||||||
outCommand({
|
|
||||||
type: OutCommand.openSettings,
|
|
||||||
data: { system_id: solar_system_id.toString() },
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
const showHandlers = isConnecting || hoverNodeId === id;
|
|
||||||
|
|
||||||
const space = showKSpaceBG ? REGIONS_MAP[region_id] : '';
|
|
||||||
const regionClass = showKSpaceBG ? SpaceToClass[space] : null;
|
|
||||||
|
|
||||||
const computedTemporaryName = useMemo(() => {
|
|
||||||
if (!isTempSystemNameEnabled) {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
if (isShowLinkedSigIdTempName && linkedSigPrefix) {
|
|
||||||
return temporary_name ? `${linkedSigPrefix}・${temporary_name}` : `${linkedSigPrefix}・${solar_system_name}`;
|
|
||||||
}
|
|
||||||
return temporary_name;
|
|
||||||
}, [isShowLinkedSigIdTempName, isTempSystemNameEnabled, linkedSigPrefix, solar_system_name, temporary_name]);
|
|
||||||
|
|
||||||
const systemName = useMemo(() => {
|
|
||||||
if (isTempSystemNameEnabled && computedTemporaryName) {
|
|
||||||
return computedTemporaryName;
|
|
||||||
}
|
|
||||||
return solar_system_name;
|
|
||||||
}, [isTempSystemNameEnabled, solar_system_name, computedTemporaryName]);
|
|
||||||
|
|
||||||
const customName = useMemo(() => {
|
|
||||||
if (isTempSystemNameEnabled && computedTemporaryName && name) {
|
|
||||||
return name;
|
|
||||||
}
|
|
||||||
if (solar_system_name !== name && name) {
|
|
||||||
return name;
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}, [isTempSystemNameEnabled, computedTemporaryName, name, solar_system_name]);
|
|
||||||
|
|
||||||
const [unsplashedLeft, unsplashedRight] = useMemo(() => {
|
|
||||||
if (!isShowUnsplashedSignatures) {
|
|
||||||
return [[], []];
|
|
||||||
}
|
|
||||||
return prepareUnsplashedChunks(
|
|
||||||
systemSigs
|
|
||||||
.filter(s => s.group === 'Wormhole' && !s.linked_system)
|
|
||||||
.map(s => ({
|
|
||||||
eve_id: s.eve_id,
|
|
||||||
type: s.type,
|
|
||||||
custom_info: s.custom_info,
|
|
||||||
kind: s.kind,
|
|
||||||
name: s.name,
|
|
||||||
group: s.group,
|
|
||||||
})) as UnsplashedSignatureType[],
|
|
||||||
);
|
|
||||||
}, [isShowUnsplashedSignatures, systemSigs]);
|
|
||||||
|
|
||||||
// Ensure hubs are always strings.
|
|
||||||
const hubsAsStrings = useMemo(() => hubs.map(item => item.toString()), [hubs]);
|
|
||||||
|
|
||||||
const nodeVars: SolarSystemNodeVars = {
|
|
||||||
id,
|
|
||||||
selected,
|
|
||||||
visible,
|
|
||||||
isWormhole,
|
|
||||||
classTitleColor,
|
|
||||||
killsCount,
|
|
||||||
killsActivityType,
|
|
||||||
hasUserCharacters,
|
|
||||||
userCharacters,
|
|
||||||
showHandlers,
|
|
||||||
regionClass,
|
|
||||||
systemName,
|
|
||||||
customName,
|
|
||||||
labelCustom,
|
|
||||||
isShattered: is_shattered,
|
|
||||||
tag,
|
|
||||||
status,
|
|
||||||
labelsInfo,
|
|
||||||
dbClick,
|
|
||||||
sortedStatics,
|
|
||||||
effectName: effect_name,
|
|
||||||
regionName: region_name,
|
|
||||||
solarSystemId: solar_system_id.toString(),
|
|
||||||
solarSystemName: solar_system_name,
|
|
||||||
locked,
|
|
||||||
hubs: hubsAsStrings,
|
|
||||||
name: name,
|
|
||||||
isConnecting,
|
|
||||||
hoverNodeId,
|
|
||||||
charactersInSystem,
|
|
||||||
unsplashedLeft,
|
|
||||||
unsplashedRight,
|
|
||||||
isThickConnections,
|
|
||||||
classTitle: class_title,
|
|
||||||
temporaryName: computedTemporaryName,
|
|
||||||
};
|
|
||||||
|
|
||||||
return nodeVars;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface SolarSystemNodeVars {
|
export interface SolarSystemNodeVars {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -250,8 +21,6 @@ export interface SolarSystemNodeVars {
|
|||||||
visible: boolean;
|
visible: boolean;
|
||||||
isWormhole: boolean;
|
isWormhole: boolean;
|
||||||
classTitleColor: string | null;
|
classTitleColor: string | null;
|
||||||
killsCount: number | null;
|
|
||||||
killsActivityType: string | null;
|
|
||||||
hasUserCharacters: boolean;
|
hasUserCharacters: boolean;
|
||||||
showHandlers: boolean;
|
showHandlers: boolean;
|
||||||
regionClass: string | null;
|
regionClass: string | null;
|
||||||
@@ -278,28 +47,192 @@ export interface SolarSystemNodeVars {
|
|||||||
unsplashedLeft: Array<SystemSignature>;
|
unsplashedLeft: Array<SystemSignature>;
|
||||||
unsplashedRight: Array<SystemSignature>;
|
unsplashedRight: Array<SystemSignature>;
|
||||||
isThickConnections: boolean;
|
isThickConnections: boolean;
|
||||||
|
isRally: boolean;
|
||||||
classTitle: string | null;
|
classTitle: string | null;
|
||||||
temporaryName?: string | null;
|
temporaryName?: string | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useNodeKillsCount(systemId: number | string, initialKillsCount: number | null): number | null {
|
const SpaceToClass: Record<string, string> = {
|
||||||
const [killsCount, setKillsCount] = useState<number | null>(initialKillsCount);
|
[Spaces.Caldari]: 'Caldaria',
|
||||||
|
[Spaces.Matar]: 'Mataria',
|
||||||
|
[Spaces.Amarr]: 'Amarria',
|
||||||
|
[Spaces.Gallente]: 'Gallente',
|
||||||
|
[Spaces.Pochven]: 'Pochven',
|
||||||
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
export function useLocalCounter(nodeVars: SolarSystemNodeVars) {
|
||||||
setKillsCount(initialKillsCount);
|
const localCounterCharacters = useMemo(() => {
|
||||||
}, [initialKillsCount]);
|
return nodeVars.charactersInSystem
|
||||||
|
.map(char => ({
|
||||||
|
...char,
|
||||||
|
compact: true,
|
||||||
|
isOwn: nodeVars.userCharacters.includes(char.eve_id),
|
||||||
|
}))
|
||||||
|
.sort((a, b) => a.name.localeCompare(b.name));
|
||||||
|
}, [nodeVars.charactersInSystem, nodeVars.userCharacters]);
|
||||||
|
return { localCounterCharacters };
|
||||||
|
}
|
||||||
|
|
||||||
useMapEventListener(event => {
|
export const useSolarSystemNode = (props: NodeProps<MapSolarSystemType>): SolarSystemNodeVars => {
|
||||||
if (event.name === Commands.killsUpdated && event.data?.toString() === systemId.toString()) {
|
const { id, data, selected } = props;
|
||||||
//@ts-ignore
|
const {
|
||||||
if (event.payload && typeof event.payload.kills === 'number') {
|
id: solar_system_id,
|
||||||
// @ts-ignore
|
locked,
|
||||||
setKillsCount(event.payload.kills);
|
name,
|
||||||
}
|
tag,
|
||||||
return true;
|
status,
|
||||||
}
|
labels,
|
||||||
return false;
|
temporary_name,
|
||||||
|
linked_sig_eve_id: linkedSigEveId = '',
|
||||||
|
} = data;
|
||||||
|
|
||||||
|
const {
|
||||||
|
storedSettings: { interfaceSettings },
|
||||||
|
data: { systemSignatures: mapSystemSignatures },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const systemStaticInfo = useMemo(() => {
|
||||||
|
return getSystemStaticInfo(solar_system_id)!;
|
||||||
|
}, [solar_system_id]);
|
||||||
|
|
||||||
|
const {
|
||||||
|
system_class,
|
||||||
|
security,
|
||||||
|
class_title,
|
||||||
|
statics,
|
||||||
|
effect_name,
|
||||||
|
region_name,
|
||||||
|
region_id,
|
||||||
|
is_shattered,
|
||||||
|
solar_system_name,
|
||||||
|
constellation_name,
|
||||||
|
} = systemStaticInfo;
|
||||||
|
|
||||||
|
const { isShowUnsplashedSignatures } = interfaceSettings;
|
||||||
|
const isTempSystemNameEnabled = useMapGetOption('show_temp_system_name') === 'true';
|
||||||
|
const isShowLinkedSigId = useMapGetOption('show_linked_signature_id') === 'true';
|
||||||
|
const isShowLinkedSigIdTempName = useMapGetOption('show_linked_signature_id_temp_name') === 'true';
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: {
|
||||||
|
characters,
|
||||||
|
wormholesData,
|
||||||
|
hubs,
|
||||||
|
userCharacters,
|
||||||
|
isConnecting,
|
||||||
|
hoverNodeId,
|
||||||
|
visibleNodes,
|
||||||
|
showKSpaceBG,
|
||||||
|
isThickConnections,
|
||||||
|
pings,
|
||||||
|
},
|
||||||
|
outCommand,
|
||||||
|
} = useMapState();
|
||||||
|
|
||||||
|
const visible = useMemo(() => visibleNodes.has(id), [id, visibleNodes]);
|
||||||
|
|
||||||
|
const systemSigs = useMemo(() => mapSystemSignatures[solar_system_id] || [], [solar_system_id, mapSystemSignatures]);
|
||||||
|
|
||||||
|
const charactersInSystem = useMemo(() => {
|
||||||
|
return characters.filter(c => c.location?.solar_system_id === parseInt(solar_system_id) && c.online);
|
||||||
|
}, [characters, solar_system_id]);
|
||||||
|
|
||||||
|
const isWormhole = isWormholeSpace(system_class);
|
||||||
|
|
||||||
|
const classTitleColor = useMemo(
|
||||||
|
() => getSystemClassStyles({ systemClass: system_class, security }),
|
||||||
|
[security, system_class],
|
||||||
|
);
|
||||||
|
|
||||||
|
const sortedStatics = useMemo(() => sortWHClasses(wormholesData, statics), [wormholesData, statics]);
|
||||||
|
|
||||||
|
const linkedSigPrefix = useMemo(() => (linkedSigEveId ? linkedSigEveId.split('-')[0] : null), [linkedSigEveId]);
|
||||||
|
|
||||||
|
const { labelsInfo, labelCustom } = useLabelsInfo({
|
||||||
|
labels,
|
||||||
|
linkedSigPrefix,
|
||||||
|
isShowLinkedSigId,
|
||||||
});
|
});
|
||||||
|
|
||||||
return killsCount;
|
const hasUserCharacters = useMemo(
|
||||||
}
|
() => charactersInSystem.some(x => userCharacters.includes(x.eve_id)),
|
||||||
|
[charactersInSystem, userCharacters],
|
||||||
|
);
|
||||||
|
|
||||||
|
const dbClick = useDoubleClick(() => {
|
||||||
|
outCommand({
|
||||||
|
type: OutCommand.openSettings,
|
||||||
|
data: { system_id: solar_system_id },
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const showHandlers = isConnecting || hoverNodeId === id;
|
||||||
|
|
||||||
|
const space = showKSpaceBG ? REGIONS_MAP[region_id] : '';
|
||||||
|
const regionClass = showKSpaceBG ? SpaceToClass[space] || null : null;
|
||||||
|
|
||||||
|
const { systemName, computedTemporaryName, customName } = useSystemName({
|
||||||
|
isTempSystemNameEnabled,
|
||||||
|
temporary_name,
|
||||||
|
isShowLinkedSigIdTempName,
|
||||||
|
linkedSigPrefix,
|
||||||
|
name,
|
||||||
|
systemStaticInfo,
|
||||||
|
});
|
||||||
|
|
||||||
|
const { unsplashedLeft, unsplashedRight } = useUnsplashedSignatures(systemSigs, isShowUnsplashedSignatures);
|
||||||
|
|
||||||
|
const hubsAsStrings = useMemo(() => hubs.map(item => item.toString()), [hubs]);
|
||||||
|
|
||||||
|
const isRally = useMemo(
|
||||||
|
() => !!pings.find(x => x.solar_system_id === solar_system_id && x.type === PingType.Rally),
|
||||||
|
[pings, solar_system_id],
|
||||||
|
);
|
||||||
|
|
||||||
|
const regionName = useMemo(() => {
|
||||||
|
if (region_id === Regions.Pochven) {
|
||||||
|
return constellation_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
return region_name;
|
||||||
|
}, [constellation_name, region_id, region_name]);
|
||||||
|
|
||||||
|
const nodeVars: SolarSystemNodeVars = {
|
||||||
|
id,
|
||||||
|
selected,
|
||||||
|
visible,
|
||||||
|
isWormhole,
|
||||||
|
classTitleColor,
|
||||||
|
hasUserCharacters,
|
||||||
|
userCharacters,
|
||||||
|
showHandlers,
|
||||||
|
regionClass,
|
||||||
|
systemName,
|
||||||
|
customName,
|
||||||
|
labelCustom,
|
||||||
|
isShattered: is_shattered,
|
||||||
|
tag,
|
||||||
|
status,
|
||||||
|
labelsInfo,
|
||||||
|
dbClick,
|
||||||
|
sortedStatics,
|
||||||
|
effectName: effect_name,
|
||||||
|
solarSystemId: solar_system_id.toString(),
|
||||||
|
locked,
|
||||||
|
hubs: hubsAsStrings,
|
||||||
|
name,
|
||||||
|
isConnecting,
|
||||||
|
hoverNodeId,
|
||||||
|
charactersInSystem,
|
||||||
|
unsplashedLeft,
|
||||||
|
unsplashedRight,
|
||||||
|
isThickConnections,
|
||||||
|
classTitle: class_title,
|
||||||
|
temporaryName: computedTemporaryName,
|
||||||
|
regionName,
|
||||||
|
solarSystemName: solar_system_name,
|
||||||
|
isRally,
|
||||||
|
};
|
||||||
|
|
||||||
|
return nodeVars;
|
||||||
|
};
|
||||||
56
assets/js/hooks/Mapper/components/map/hooks/useSystemName.ts
Normal file
56
assets/js/hooks/Mapper/components/map/hooks/useSystemName.ts
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
|
||||||
|
import { useMemo } from 'react';
|
||||||
|
|
||||||
|
interface UseSystemNameParams {
|
||||||
|
isTempSystemNameEnabled: boolean;
|
||||||
|
temporary_name?: string | null;
|
||||||
|
isShowLinkedSigIdTempName: boolean;
|
||||||
|
linkedSigPrefix: string | null;
|
||||||
|
name?: string | null;
|
||||||
|
systemStaticInfo: SolarSystemStaticInfoRaw;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useSystemName = ({
|
||||||
|
isTempSystemNameEnabled,
|
||||||
|
temporary_name,
|
||||||
|
isShowLinkedSigIdTempName,
|
||||||
|
linkedSigPrefix,
|
||||||
|
name,
|
||||||
|
systemStaticInfo,
|
||||||
|
}: UseSystemNameParams) => {
|
||||||
|
const { solar_system_name = '' } = systemStaticInfo;
|
||||||
|
|
||||||
|
const computedTemporaryName = useMemo(() => {
|
||||||
|
if (!isTempSystemNameEnabled) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isShowLinkedSigIdTempName && linkedSigPrefix) {
|
||||||
|
return temporary_name ? `${linkedSigPrefix}:${temporary_name}` : `${linkedSigPrefix}:${solar_system_name}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return temporary_name ?? '';
|
||||||
|
}, [isTempSystemNameEnabled, temporary_name, solar_system_name, isShowLinkedSigIdTempName, linkedSigPrefix]);
|
||||||
|
|
||||||
|
const systemName = useMemo(() => {
|
||||||
|
if (isTempSystemNameEnabled && computedTemporaryName) {
|
||||||
|
return computedTemporaryName;
|
||||||
|
}
|
||||||
|
|
||||||
|
return solar_system_name;
|
||||||
|
}, [isTempSystemNameEnabled, computedTemporaryName, solar_system_name]);
|
||||||
|
|
||||||
|
const customName = useMemo(() => {
|
||||||
|
if (isTempSystemNameEnabled && computedTemporaryName && name) {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (solar_system_name !== name && name) {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}, [isTempSystemNameEnabled, computedTemporaryName, name, solar_system_name]);
|
||||||
|
|
||||||
|
return { systemName, computedTemporaryName, customName };
|
||||||
|
};
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
import { useMemo } from 'react';
|
||||||
|
import { SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { prepareUnsplashedChunks } from '@/hooks/Mapper/components/map/helpers';
|
||||||
|
|
||||||
|
export type UnsplashedSignatureType = SystemSignature & { sig_id: string };
|
||||||
|
|
||||||
|
export function useUnsplashedSignatures(systemSigs: SystemSignature[], isShowUnsplashedSignatures: boolean) {
|
||||||
|
return useMemo(() => {
|
||||||
|
if (!isShowUnsplashedSignatures) {
|
||||||
|
return {
|
||||||
|
unsplashedLeft: [] as SystemSignature[],
|
||||||
|
unsplashedRight: [] as SystemSignature[],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
const chunks = prepareUnsplashedChunks(
|
||||||
|
systemSigs
|
||||||
|
.filter(s => s.group === 'Wormhole' && !s.linked_system)
|
||||||
|
.map(s => ({
|
||||||
|
eve_id: s.eve_id,
|
||||||
|
type: s.type,
|
||||||
|
custom_info: s.custom_info,
|
||||||
|
kind: s.kind,
|
||||||
|
name: s.name,
|
||||||
|
group: s.group,
|
||||||
|
})) as UnsplashedSignatureType[],
|
||||||
|
);
|
||||||
|
const [unsplashedLeft, unsplashedRight] = chunks;
|
||||||
|
return { unsplashedLeft, unsplashedRight };
|
||||||
|
}, [isShowUnsplashedSignatures, systemSigs]);
|
||||||
|
}
|
||||||
@@ -1,14 +1,14 @@
|
|||||||
import { useCallback, useEffect, useRef } from 'react';
|
|
||||||
import { Node, useOnViewportChange, useReactFlow } from 'reactflow';
|
|
||||||
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
||||||
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
|
import { useCallback, useEffect, useRef } from 'react';
|
||||||
|
import { Node, useOnViewportChange, useReactFlow } from 'reactflow';
|
||||||
|
|
||||||
const useThrottle = () => {
|
const useThrottle = () => {
|
||||||
const throttleSeed = useRef<number | null>(null);
|
const throttleSeed = useRef<number | null>(null);
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
const throttleFunction = useRef((func: any, delay = 200) => {
|
const throttleFunction = useRef((func: any, delay = 200) => {
|
||||||
if (!throttleSeed.current) {
|
if (!throttleSeed.current) {
|
||||||
// Call the callback immediately for the first time
|
|
||||||
func();
|
func();
|
||||||
throttleSeed.current = setTimeout(() => {
|
throttleSeed.current = setTimeout(() => {
|
||||||
throttleSeed.current = null;
|
throttleSeed.current = null;
|
||||||
@@ -75,7 +75,7 @@ export const useUpdateNodes = (nodes: Node<SolarSystemRawType>[]) => {
|
|||||||
|
|
||||||
const visibleNodes = new Set(nodes.filter(x => isNodeVisible(x, viewport)).map(x => x.id));
|
const visibleNodes = new Set(nodes.filter(x => isNodeVisible(x, viewport)).map(x => x.id));
|
||||||
update({ visibleNodes });
|
update({ visibleNodes });
|
||||||
}, [nodes]);
|
}, [getViewport, nodes, update]);
|
||||||
|
|
||||||
useOnViewportChange({
|
useOnViewportChange({
|
||||||
onChange: () => throttle(updateNodesVisibility.bind(this)),
|
onChange: () => throttle(updateNodesVisibility.bind(this)),
|
||||||
@@ -84,5 +84,5 @@ export const useUpdateNodes = (nodes: Node<SolarSystemRawType>[]) => {
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
updateNodesVisibility();
|
updateNodesVisibility();
|
||||||
}, [nodes]);
|
}, [nodes, updateNodesVisibility]);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
@import './eve-common-variables';
|
@use './eve-common-variables';
|
||||||
@import './eve-common';
|
@use './eve-common';
|
||||||
|
|
||||||
.default-theme {
|
.default-theme {
|
||||||
--rf-bg-color: #0C0A09;
|
--rf-bg-color: #0C0A09;
|
||||||
|
|||||||
@@ -1,18 +1,19 @@
|
|||||||
|
@use "sass:color";
|
||||||
|
|
||||||
$friendlyBase: #3bbd39;
|
$friendlyBase: #3bbd39;
|
||||||
$friendlyAlpha: #3bbd3952;
|
$friendlyAlpha: #3bbd3952;
|
||||||
$friendlyDark20: darken($friendlyBase, 20%);
|
$friendlyDark20: color.adjust($friendlyBase, $lightness: -20%);
|
||||||
$friendlyDark30: darken($friendlyBase, 30%);
|
$friendlyDark30: color.adjust($friendlyBase, $lightness: -30%);
|
||||||
$friendlyDark5: darken($friendlyBase, 5%);
|
$friendlyDark5: color.adjust($friendlyBase, $lightness: -5%);
|
||||||
|
|
||||||
$lookingForBase: #43c2fd;
|
$lookingForBase: #43c2fd;
|
||||||
$lookingForAlpha: rgba(67, 176, 253, 0.48);
|
$lookingForAlpha: rgba(67, 176, 253, 0.48);
|
||||||
$lookingForDark15: darken($lookingForBase, 15%);
|
$lookingForDark15: color.adjust($lookingForBase, $lightness: -15%);
|
||||||
|
|
||||||
$homeBase: rgb(179, 253, 67);
|
$homeBase: rgb(179, 253, 67);
|
||||||
$homeAlpha: rgba(186, 248, 48, 0.32);
|
$homeAlpha: rgba(186, 248, 48, 0.32);
|
||||||
$homeBackground: #a0fa5636;
|
$homeBackground: #a0fa5636;
|
||||||
$homeDark30: darken($homeBase, 30%);
|
$homeDark30: color.adjust($homeBase, $lightness: -30%);
|
||||||
|
|
||||||
:root {
|
:root {
|
||||||
--pastel-blue: #5a7d9a;
|
--pastel-blue: #5a7d9a;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
@import './eve-common-variables';
|
@use './eve-common-variables';
|
||||||
|
|
||||||
|
|
||||||
.eve-wh-effect-color-pulsar {
|
.eve-wh-effect-color-pulsar {
|
||||||
@@ -542,48 +542,32 @@
|
|||||||
background-color: #d10600;
|
background-color: #d10600;
|
||||||
}
|
}
|
||||||
|
|
||||||
.react-flow {
|
|
||||||
color: var(--text-color);
|
|
||||||
|
|
||||||
&__pane {
|
.react-flow__minimap-node {
|
||||||
cursor: auto;
|
fill: #ffb03a;
|
||||||
}
|
}
|
||||||
|
|
||||||
&__minimap {
|
.react-flow__minimap {
|
||||||
background-color: rgba(66, 66, 66, 1);
|
border: 1px solid #282828;
|
||||||
opacity: 0.7;
|
border-radius: 4px;
|
||||||
border: 1px solid #2f2f2f;
|
background-color: rgb(47 37 37) !important;
|
||||||
border-radius: 4px;
|
overflow: hidden;
|
||||||
overflow: hidden;
|
}
|
||||||
}
|
|
||||||
|
|
||||||
&__minimap-mask {
|
.react-flow__minimap-mask {
|
||||||
fill: rgba(28, 28, 28, 0.75);
|
stroke-width: 2px;
|
||||||
}
|
fill: rgba(0, 0, 0, 0.5);
|
||||||
|
mix-blend-mode: overlay;
|
||||||
|
}
|
||||||
|
|
||||||
&__controls {
|
.react-flow__minimap-mask {
|
||||||
filter: brightness(1.5);
|
stroke-width: 2px;
|
||||||
}
|
fill: rgb(0 0 0 / 50%) !important;
|
||||||
|
mix-blend-mode: inherit;
|
||||||
&__minimap-node {
|
opacity: 1;
|
||||||
fill: #ffb03a;
|
stroke: #fff;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.context-menu-active {
|
.context-menu-active {
|
||||||
background-color: rgba(131, 131, 131, 0.33);
|
background-color: rgba(131, 131, 131, 0.33);
|
||||||
}
|
}
|
||||||
|
|
||||||
.p-dialog {
|
|
||||||
.p-dialog-header {
|
|
||||||
height: 40px;
|
|
||||||
padding: 1rem;
|
|
||||||
padding-right: 10px !important;
|
|
||||||
}
|
|
||||||
.p-dialog-title {
|
|
||||||
font-size: 1rem !important;
|
|
||||||
}
|
|
||||||
.p-dialog-header-icons {
|
|
||||||
align-self: initial !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,2 +1,2 @@
|
|||||||
@import './default-theme.scss';
|
@use './default-theme.scss';
|
||||||
@import './pathfinder-theme.scss';
|
@use './pathfinder-theme.scss';
|
||||||
@@ -1,10 +1,11 @@
|
|||||||
@import './eve-common-variables';
|
@use "sass:color";
|
||||||
@import './eve-common';
|
@use './eve-common-variables';
|
||||||
|
@use './eve-common';
|
||||||
@import url('https://fonts.googleapis.com/css2?family=Oxygen:wght@300;400;700&display=swap');
|
@import url('https://fonts.googleapis.com/css2?family=Oxygen:wght@300;400;700&display=swap');
|
||||||
|
|
||||||
$homeBase: rgb(197, 253, 67);
|
$homeBase: rgb(197, 253, 67);
|
||||||
$homeAlpha: rgba(197, 253, 67, 0.32);
|
$homeAlpha: rgba(197, 253, 67, 0.32);
|
||||||
$homeDark30: darken($homeBase, 30%);
|
$homeDark30: color.adjust($homeBase, $lightness: -30%);
|
||||||
|
|
||||||
.pathfinder-theme {
|
.pathfinder-theme {
|
||||||
/* -- Override values from the default theme -- */
|
/* -- Override values from the default theme -- */
|
||||||
|
|||||||
@@ -1,37 +1,48 @@
|
|||||||
import { Position, internalsSymbol } from 'reactflow';
|
import { Position, internalsSymbol, Node } from 'reactflow';
|
||||||
|
|
||||||
// returns the position (top,right,bottom or right) passed node compared to
|
type Coords = [number, number];
|
||||||
function getParams(nodeA, nodeB) {
|
type CoordsWithPosition = [number, number, Position];
|
||||||
|
|
||||||
|
function segmentsIntersect(a1: number, a2: number, b1: number, b2: number): boolean {
|
||||||
|
const [minA, maxA] = a1 < a2 ? [a1, a2] : [a2, a1];
|
||||||
|
const [minB, maxB] = b1 < b2 ? [b1, b2] : [b2, b1];
|
||||||
|
|
||||||
|
return maxA >= minB && maxB >= minA;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getParams(nodeA: Node, nodeB: Node): CoordsWithPosition {
|
||||||
const centerA = getNodeCenter(nodeA);
|
const centerA = getNodeCenter(nodeA);
|
||||||
const centerB = getNodeCenter(nodeB);
|
const centerB = getNodeCenter(nodeB);
|
||||||
|
|
||||||
const horizontalDiff = Math.abs(centerA.x - centerB.x);
|
|
||||||
const verticalDiff = Math.abs(centerA.y - centerB.y);
|
|
||||||
|
|
||||||
let position: Position;
|
let position: Position;
|
||||||
|
|
||||||
// when the horizontal difference between the nodes is bigger, we use Position.Left or Position.Right for the handle
|
if (
|
||||||
if (horizontalDiff > verticalDiff) {
|
segmentsIntersect(
|
||||||
position = centerA.x > centerB.x ? Position.Left : Position.Right;
|
nodeA.positionAbsolute!.x - 10,
|
||||||
} else {
|
nodeA.positionAbsolute!.x - 10 + nodeA.width! + 20,
|
||||||
// here the vertical difference between the nodes is bigger, so we use Position.Top or Position.Bottom for the handle
|
nodeB.positionAbsolute!.x,
|
||||||
|
nodeB.positionAbsolute!.x + nodeB.width!,
|
||||||
|
)
|
||||||
|
) {
|
||||||
position = centerA.y > centerB.y ? Position.Top : Position.Bottom;
|
position = centerA.y > centerB.y ? Position.Top : Position.Bottom;
|
||||||
|
} else {
|
||||||
|
position = centerA.x > centerB.x ? Position.Left : Position.Right;
|
||||||
}
|
}
|
||||||
|
|
||||||
const [x, y] = getHandleCoordsByPosition(nodeA, position);
|
const [x, y] = getHandleCoordsByPosition(nodeA, position);
|
||||||
return [x, y, position];
|
return [x, y, position];
|
||||||
}
|
}
|
||||||
|
|
||||||
function getHandleCoordsByPosition(node, handlePosition) {
|
function getHandleCoordsByPosition(node: Node, handlePosition: Position): Coords {
|
||||||
// all handles are from type source, that's why we use handleBounds.source here
|
const handle = node[internalsSymbol]!.handleBounds!.source!.find(h => h.position === handlePosition);
|
||||||
const handle = node[internalsSymbol].handleBounds.source.find(h => h.position === handlePosition);
|
|
||||||
|
if (!handle) {
|
||||||
|
throw new Error(`Handle with position ${handlePosition} not found on node ${node.id}`);
|
||||||
|
}
|
||||||
|
|
||||||
let offsetX = handle.width / 2;
|
let offsetX = handle.width / 2;
|
||||||
let offsetY = handle.height / 2;
|
let offsetY = handle.height / 2;
|
||||||
|
|
||||||
// this is a tiny detail to make the markerEnd of an edge visible.
|
|
||||||
// The handle position that gets calculated has the origin top-left, so depending which side we are using, we add a little offset
|
|
||||||
// when the handlePosition is Position.Right for example, we need to add an offset as big as the handle itself in order to get the correct position
|
|
||||||
switch (handlePosition) {
|
switch (handlePosition) {
|
||||||
case Position.Left:
|
case Position.Left:
|
||||||
offsetX = 0;
|
offsetX = 0;
|
||||||
@@ -47,21 +58,20 @@ function getHandleCoordsByPosition(node, handlePosition) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
const x = node.positionAbsolute.x + handle.x + offsetX;
|
const x = node.positionAbsolute!.x + handle.x + offsetX;
|
||||||
const y = node.positionAbsolute.y + handle.y + offsetY;
|
const y = node.positionAbsolute!.y + handle.y + offsetY;
|
||||||
|
|
||||||
return [x, y];
|
return [x, y];
|
||||||
}
|
}
|
||||||
|
|
||||||
function getNodeCenter(node) {
|
function getNodeCenter(node: Node): { x: number; y: number } {
|
||||||
return {
|
return {
|
||||||
x: node.positionAbsolute.x + node.width / 2,
|
x: node.positionAbsolute!.x + node.width! / 2,
|
||||||
y: node.positionAbsolute.y + node.height / 2,
|
y: node.positionAbsolute!.y + node.height! / 2,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns the parameters (sx, sy, tx, ty, sourcePos, targetPos) you need to create an edge
|
export function getEdgeParams(source: Node, target: Node) {
|
||||||
export function getEdgeParams(source, target) {
|
|
||||||
const [sx, sy, sourcePos] = getParams(source, target);
|
const [sx, sy, sourcePos] = getParams(source, target);
|
||||||
const [tx, ty, targetPos] = getParams(target, source);
|
const [tx, ty, targetPos] = getParams(target, source);
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,52 @@
|
|||||||
|
import { MarkdownComment } from '@/hooks/Mapper/components/mapInterface/components/Comments/components';
|
||||||
|
import { useEffect, useRef, useState } from 'react';
|
||||||
|
import { CommentType } from '@/hooks/Mapper/types';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
|
export interface CommentsProps {}
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-empty-pattern
|
||||||
|
export const Comments = ({}: CommentsProps) => {
|
||||||
|
const [commentsList, setCommentsList] = useState<CommentType[]>([]);
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: { selectedSystems },
|
||||||
|
comments: { loadComments, comments, lastUpdateKey },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const [systemId] = selectedSystems;
|
||||||
|
|
||||||
|
const ref = useRef({ loadComments, systemId });
|
||||||
|
ref.current = { loadComments, systemId };
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const commentsBySystem = comments.get(systemId);
|
||||||
|
if (!commentsBySystem) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const els = [...commentsBySystem.comments].sort((a, b) => +new Date(b.updated_at) - +new Date(a.updated_at));
|
||||||
|
|
||||||
|
setCommentsList(els);
|
||||||
|
}, [systemId, lastUpdateKey, comments]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
ref.current.loadComments(systemId);
|
||||||
|
}, [systemId]);
|
||||||
|
|
||||||
|
if (commentsList.length === 0) {
|
||||||
|
return (
|
||||||
|
<div className="w-full h-full flex justify-center items-center select-none text-stone-400/80 text-sm">
|
||||||
|
Not comments found here
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col gap-1 whitespace-nowrap overflow-auto text-ellipsis custom-scrollbar">
|
||||||
|
{commentsList.map(({ id, text, updated_at, characterEveId }) => (
|
||||||
|
<MarkdownComment key={id} text={text} time={updated_at} characterEveId={characterEveId} id={id} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,70 @@
|
|||||||
|
.MarkdownCommentRoot {
|
||||||
|
border-left-width: 3px;
|
||||||
|
|
||||||
|
@apply text-[12px] leading-[1.2] text-stone-300 break-words;
|
||||||
|
@apply bg-gradient-to-r from-stone-600/40 via-stone-600/10 to-stone-600/0;
|
||||||
|
|
||||||
|
.h1 {
|
||||||
|
@apply text-[12px] font-normal m-0 p-0 border-none break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.h2 {
|
||||||
|
@apply text-[12px] font-normal m-0 p-0 border-none break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.h3 {
|
||||||
|
@apply text-[12px] font-normal m-0 p-0 border-none break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.h4 {
|
||||||
|
@apply text-[12px] font-normal m-0 p-0 border-none break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.h5 {
|
||||||
|
@apply text-[12px] font-normal m-0 p-0 border-none break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
.h6 {
|
||||||
|
@apply text-[12px] font-normal m-0 p-0 border-none break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
p {
|
||||||
|
@apply m-0 p-0 break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul, ol {
|
||||||
|
@apply m-0 p-0 list-none;
|
||||||
|
}
|
||||||
|
|
||||||
|
li {
|
||||||
|
@apply m-0 break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
blockquote {
|
||||||
|
@apply border-l-4 border-cyan-400 p-2 m-0 font-normal text-stone-300 italic break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
@apply text-violet-400 cursor-pointer transition-colors duration-200 break-words whitespace-normal;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
@apply underline;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
b, strong {
|
||||||
|
@apply font-bold text-green-400 break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
i, em {
|
||||||
|
@apply italic text-pink-400 break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
del {
|
||||||
|
@apply line-through text-stone-500 break-words whitespace-normal;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
@apply border-none h-[1px] bg-cyan-400 opacity-50 my-2;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,94 @@
|
|||||||
|
import classes from './MarkdownComment.module.scss';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import {
|
||||||
|
InfoDrawer,
|
||||||
|
MarkdownTextViewer,
|
||||||
|
TimeAgo,
|
||||||
|
TooltipPosition,
|
||||||
|
WdImgButton,
|
||||||
|
} from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { useGetCacheCharacter } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
|
import { useCallback, useRef, useState } from 'react';
|
||||||
|
import { WdTransition } from '@/hooks/Mapper/components/ui-kit/WdTransition/WdTransition.tsx';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { ConfirmPopup } from 'primereact/confirmpopup';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
import { useConfirmPopup } from '@/hooks/Mapper/hooks';
|
||||||
|
|
||||||
|
const TOOLTIP_PROPS = { content: 'Remove comment', position: TooltipPosition.top };
|
||||||
|
|
||||||
|
export interface MarkdownCommentProps {
|
||||||
|
text: string;
|
||||||
|
time: string;
|
||||||
|
characterEveId: string;
|
||||||
|
id: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const MarkdownComment = ({ text, time, characterEveId, id }: MarkdownCommentProps) => {
|
||||||
|
const char = useGetCacheCharacter(characterEveId);
|
||||||
|
const [hovered, setHovered] = useState(false);
|
||||||
|
|
||||||
|
const { cfShow, cfHide, cfVisible, cfRef } = useConfirmPopup();
|
||||||
|
|
||||||
|
const { outCommand } = useMapRootState();
|
||||||
|
const ref = useRef({ outCommand, id });
|
||||||
|
ref.current = { outCommand, id };
|
||||||
|
|
||||||
|
const handleDelete = useCallback(async () => {
|
||||||
|
await ref.current.outCommand({
|
||||||
|
type: OutCommand.deleteSystemComment,
|
||||||
|
data: ref.current.id,
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleMouseEnter = useCallback(() => setHovered(true), []);
|
||||||
|
const handleMouseLeave = useCallback(() => setHovered(false), []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<InfoDrawer
|
||||||
|
labelClassName="mb-[3px]"
|
||||||
|
className={clsx(classes.MarkdownCommentRoot, 'p-1 bg-stone-700/20 ')}
|
||||||
|
onMouseEnter={handleMouseEnter}
|
||||||
|
onMouseLeave={handleMouseLeave}
|
||||||
|
title={
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<div>
|
||||||
|
<span className="text-stone-500">
|
||||||
|
by <span className="text-orange-300/70">{char?.data?.name ?? ''}</span>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<WdTransition active={hovered} timeout={100}>
|
||||||
|
<div className="text-stone-500 max-h-[12px]">
|
||||||
|
{!hovered && <TimeAgo timestamp={time} />}
|
||||||
|
{hovered && (
|
||||||
|
// @ts-ignore
|
||||||
|
<div ref={cfRef}>
|
||||||
|
<WdImgButton
|
||||||
|
className={clsx(PrimeIcons.TRASH, 'hover:text-red-400')}
|
||||||
|
tooltip={TOOLTIP_PROPS}
|
||||||
|
onClick={cfShow}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</WdTransition>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<MarkdownTextViewer>{text}</MarkdownTextViewer>
|
||||||
|
</InfoDrawer>
|
||||||
|
|
||||||
|
<ConfirmPopup
|
||||||
|
target={cfRef.current}
|
||||||
|
visible={cfVisible}
|
||||||
|
onHide={cfHide}
|
||||||
|
message="Are you sure you want to delete?"
|
||||||
|
icon="pi pi-exclamation-triangle"
|
||||||
|
accept={handleDelete}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './MarkdownComment.tsx';
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './MarkdownComment';
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './Comments';
|
||||||
@@ -0,0 +1,26 @@
|
|||||||
|
export const markdown = `
|
||||||
|
# Heading 1
|
||||||
|
## Heading 2
|
||||||
|
### Heading 3
|
||||||
|
#### Heading 4
|
||||||
|
##### Heading 5
|
||||||
|
###### Heading 6
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Paragraphs
|
||||||
|
This is a regular text paragraph.
|
||||||
|
|
||||||
|
Another paragraph, but with **bold** and *italic* text, as well as ~~strikethrough~~.
|
||||||
|
|
||||||
|
> This is a block quote.
|
||||||
|
> Second line of the quote.
|
||||||
|
|
||||||
|
## Links
|
||||||
|
[Link to Google](https://www.google.com)
|
||||||
|
|
||||||
|
## Horizontal Line
|
||||||
|
A block quote with ~strikethrough~ and a URL: https://reactjs.org.
|
||||||
|
---
|
||||||
|
|
||||||
|
`;
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
import { TooltipPosition, WdImageSize, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { MarkdownEditor } from '@/hooks/Mapper/components/mapInterface/components/MarkdownEditor';
|
||||||
|
import { useHotkey } from '@/hooks/Mapper/hooks';
|
||||||
|
import { useCallback, useRef, useState } from 'react';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
|
export interface CommentsEditorProps {}
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-empty-pattern
|
||||||
|
export const CommentsEditor = ({}: CommentsEditorProps) => {
|
||||||
|
const [textVal, setTextVal] = useState('');
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: { selectedSystems },
|
||||||
|
outCommand,
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const [systemId] = selectedSystems;
|
||||||
|
|
||||||
|
const ref = useRef({ outCommand, systemId, textVal });
|
||||||
|
ref.current = { outCommand, systemId, textVal };
|
||||||
|
|
||||||
|
const handleFinishEdit = useCallback(async () => {
|
||||||
|
if (ref.current.textVal === '') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await ref.current.outCommand({
|
||||||
|
type: OutCommand.addSystemComment,
|
||||||
|
data: {
|
||||||
|
solarSystemId: ref.current.systemId,
|
||||||
|
value: ref.current.textVal,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
setTextVal('');
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleClick = async () => {
|
||||||
|
await handleFinishEdit();
|
||||||
|
};
|
||||||
|
|
||||||
|
useHotkey(true, ['Enter'], async () => {
|
||||||
|
await handleFinishEdit();
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<MarkdownEditor
|
||||||
|
value={textVal}
|
||||||
|
onChange={setTextVal}
|
||||||
|
overlayContent={
|
||||||
|
<div className="w-full h-full flex justify-end items-end pointer-events-none pb-[1px] pr-[8px]">
|
||||||
|
<WdImgButton
|
||||||
|
disabled={textVal.length === 0}
|
||||||
|
tooltip={{
|
||||||
|
position: TooltipPosition.bottom,
|
||||||
|
content: (
|
||||||
|
<span>
|
||||||
|
Also you may use <span className="text-cyan-400">Meta + Enter</span> hotkey.
|
||||||
|
</span>
|
||||||
|
),
|
||||||
|
}}
|
||||||
|
textSize={WdImageSize.large}
|
||||||
|
className={clsx(PrimeIcons.SEND, 'text-[14px]')}
|
||||||
|
onClick={handleClick}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user