mirror of
				https://github.com/wanderer-industries/wanderer
				synced 2025-11-04 00:14:52 +00:00 
			
		
		
		
	Compare commits
	
		
			382 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					6d99c54af7 | ||
| 
						 | 
					2b7901e9a8 | ||
| 
						 | 
					fb06dd1dbc | ||
| 
						 | 
					f8ba36b8be | ||
| 
						 | 
					5bf9d99b3d | ||
| 
						 | 
					7cad05342a | ||
| 
						 | 
					867780e525 | ||
| 
						 | 
					ff4f9a79c9 | ||
| 
						 | 
					6699c36fb3 | ||
| 
						 | 
					abd4556994 | ||
| 
						 | 
					ccf0d17371 | ||
| 
						 | 
					898584bbb6 | ||
| 
						 | 
					6d7a267e39 | ||
| 
						 | 
					9f656ca3cb | ||
| 
						 | 
					fede6451e2 | ||
| 
						 | 
					9797ad380c | ||
| 
						 | 
					33bc4a4d22 | ||
| 
						 | 
					30fc972d78 | ||
| 
						 | 
					c022b31c79 | ||
| 
						 | 
					049b06bb39 | ||
| 
						 | 
					e17d5213c0 | ||
| 
						 | 
					dcf681941e | ||
| 
						 | 
					1cd7d40405 | ||
| 
						 | 
					fbd80ba2c7 | ||
| 
						 | 
					88ab85bd04 | ||
| 
						 | 
					78f98744fd | ||
| 
						 | 
					9c9634a927 | ||
| 
						 | 
					be47be626c | ||
| 
						 | 
					2fbd3d8e19 | ||
| 
						 | 
					d5c3d4c051 | ||
| 
						 | 
					fac60f7ddd | ||
| 
						 | 
					c371478c61 | ||
| 
						 | 
					5911e29f34 | ||
| 
						 | 
					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 | ||
| 
						 | 
					6b97d36bf1 | ||
| 
						 | 
					82f6a7f701 | ||
| 
						 | 
					2d92dfbafa | ||
| 
						 | 
					f81f41f555 | ||
| 
						 | 
					54c7b44d69 | ||
| 
						 | 
					9da6605ccb | ||
| 
						 | 
					a90bf9762a | ||
| 
						 | 
					c87cfb3c43 | ||
| 
						 | 
					85cb9ccfa8 | ||
| 
						 | 
					da2639786d | ||
| 
						 | 
					3cf77da293 | ||
| 
						 | 
					3dd7633194 | ||
| 
						 | 
					ae7f4edf4a | ||
| 
						 | 
					52eab28f27 | ||
| 
						 | 
					6098d32bce | ||
| 
						 | 
					1839834771 | ||
| 
						 | 
					7cdfb87853 | ||
| 
						 | 
					3d54783a3e | ||
| 
						 | 
					f965461820 | ||
| 
						 | 
					6d67f87d4b | ||
| 
						 | 
					60697a50c2 | ||
| 
						 | 
					778d23da06 | ||
| 
						 | 
					0ee9a15d5d | ||
| 
						 | 
					24bb902bb9 | ||
| 
						 | 
					32fe6395a1 | ||
| 
						 | 
					5f506bf4b2 | ||
| 
						 | 
					0127ebfe46 | ||
| 
						 | 
					8c5366fd9b | ||
| 
						 | 
					dbcad892a9 | ||
| 
						 | 
					6da3096db1 | ||
| 
						 | 
					cd8efcd6e3 | ||
| 
						 | 
					b52471ae5e | ||
| 
						 | 
					438fecb61f | ||
| 
						 | 
					70b589a359 | ||
| 
						 | 
					cf7069b3b2 | ||
| 
						 | 
					b2198e469e | ||
| 
						 | 
					8ab337e8e7 | ||
| 
						 | 
					51878ab503 | ||
| 
						 | 
					401dfad298 | ||
| 
						 | 
					18cff7d312 | ||
| 
						 | 
					7896de00d6 | ||
| 
						 | 
					3b079505c3 | ||
| 
						 | 
					5b972b03e5 | ||
| 
						 | 
					79b284c46d | ||
| 
						 | 
					b29e57b3a4 | ||
| 
						 | 
					c6f4baeee3 | ||
| 
						 | 
					6d341be072 | ||
| 
						 | 
					2437ec9c84 | ||
| 
						 | 
					7e692b5805 | ||
| 
						 | 
					01b7370ecd | ||
| 
						 | 
					20ad8b07d7 | ||
| 
						 | 
					cab1880fb0 | ||
| 
						 | 
					78eefcd6a7 | ||
| 
						 | 
					eec78d38a8 | ||
| 
						 | 
					73f8b1f06b | ||
| 
						 | 
					f96cb01860 | ||
| 
						 | 
					6800be1bb6 | ||
| 
						 | 
					143f0a5b3a | ||
| 
						 | 
					b6495504f8 | ||
| 
						 | 
					2f07ec1b74 | ||
| 
						 | 
					7073a0e8e6 | ||
| 
						 | 
					bb0d91a3c7 | ||
| 
						 | 
					1cb12b97ba | ||
| 
						 | 
					860d20dc66 | ||
| 
						 | 
					a850071965 | ||
| 
						 | 
					fc41573e70 | ||
| 
						 | 
					97f1808fb5 | ||
| 
						 | 
					d31046eebb | ||
| 
						 | 
					a70fa50eab | ||
| 
						 | 
					9a082c26f5 | ||
| 
						 | 
					6af2dc1ed5 | ||
| 
						 | 
					5fd1509d44 | ||
| 
						 | 
					2448c0531b | ||
| 
						 | 
					b685ea1013 | ||
| 
						 | 
					55465688c8 | ||
| 
						 | 
					ac3c7e0c44 | ||
| 
						 | 
					2d6ab5646c | ||
| 
						 | 
					67b373ac29 | ||
| 
						 | 
					678169e6fa | ||
| 
						 | 
					7ee3c8db82 | ||
| 
						 | 
					304f4b01ab | ||
| 
						 | 
					4af12c21b2 | ||
| 
						 | 
					497da1e5f7 | ||
| 
						 | 
					5bd968acae | ||
| 
						 | 
					f74c20142c | ||
| 
						 | 
					d4c40d7542 | ||
| 
						 | 
					04f3fec0c0 | ||
| 
						 | 
					cd0b4b0fc9 | ||
| 
						 | 
					e7b115e6e6 | ||
| 
						 | 
					dff8fc6396 | ||
| 
						 | 
					afdaeb3d34 | ||
| 
						 | 
					ac6053361e | ||
| 
						 | 
					eb3e1ba3aa | ||
| 
						 | 
					8468a9b5de | ||
| 
						 | 
					5eafe59dcb | ||
| 
						 | 
					b38bcaa8cf | ||
| 
						 | 
					8a238a447d | ||
| 
						 | 
					3731219216 | ||
| 
						 | 
					73d5fd5f67 | ||
| 
						 | 
					e8e4aed6d5 | ||
| 
						 | 
					63571a462f | ||
| 
						 | 
					606add4142 | ||
| 
						 | 
					dac480b059 | ||
| 
						 | 
					5f67cb1dd7 | ||
| 
						 | 
					5886fff753 | ||
| 
						 | 
					da2e12bdd1 | ||
| 
						 | 
					05c3d20e56 | ||
| 
						 | 
					4633d26517 | ||
| 
						 | 
					30b0556d47 | ||
| 
						 | 
					e094378dc5 | ||
| 
						 | 
					0c48189503 | ||
| 
						 | 
					a5c346627a | ||
| 
						 | 
					4e526040bf | ||
| 
						 | 
					869c25cd60 | ||
| 
						 | 
					6aac698cd8 | ||
| 
						 | 
					230016b90f | ||
| 
						 | 
					4b1aef8dd9 | ||
| 
						 | 
					d34509d7a0 | ||
| 
						 | 
					fca98ec232 | ||
| 
						 | 
					e2814e95bd | ||
| 
						 | 
					68a3f84704 | ||
| 
						 | 
					4bc76feefc | ||
| 
						 | 
					da39a55fd0 | ||
| 
						 | 
					ee3cf04cd4 | ||
| 
						 | 
					d79e7fe2ff | ||
| 
						 | 
					8de9fdef32 | ||
| 
						 | 
					f51deeec2d | ||
| 
						 | 
					a971c69a96 | ||
| 
						 | 
					b7995f50de | ||
| 
						 | 
					14997a2959 | ||
| 
						 | 
					8fef6bcf82 | ||
| 
						 | 
					1f82d23963 | ||
| 
						 | 
					28317a2431 | ||
| 
						 | 
					6aac496a57 | ||
| 
						 | 
					ac9306b713 | ||
| 
						 | 
					d55e804efa | ||
| 
						 | 
					08407a5679 | ||
| 
						 | 
					c37d175bec | ||
| 
						 | 
					69c5326e72 | ||
| 
						 | 
					305f63e11d | ||
| 
						 | 
					698fd5e083 | ||
| 
						 | 
					1af8342d30 | ||
| 
						 | 
					68b59da78e | ||
| 
						 | 
					e784a3f850 | ||
| 
						 | 
					a45e2f3fc2 | ||
| 
						 | 
					8a3d920c31 | ||
| 
						 | 
					996d7c47bd | ||
| 
						 | 
					8d2b9db430 | ||
| 
						 | 
					423ce343c7 | ||
| 
						 | 
					1c17912d9f | ||
| 
						 | 
					6714eb5d9b | ||
| 
						 | 
					1620e1fd21 | ||
| 
						 | 
					859014874f | ||
| 
						 | 
					ef44881f06 | ||
| 
						 | 
					b0532325fa | ||
| 
						 | 
					2c00bd426e | ||
| 
						 | 
					6eccf2ac67 | ||
| 
						 | 
					973a1e54b3 | ||
| 
						 | 
					2b42b637df | ||
| 
						 | 
					b950572818 | ||
| 
						 | 
					e470a210f1 | ||
| 
						 | 
					71ec2d413c | ||
| 
						 | 
					9122412558 | ||
| 
						 | 
					0ba5c963b4 | ||
| 
						 | 
					39a0ce284f | ||
| 
						 | 
					f9d580dbc0 | ||
| 
						 | 
					5c41574328 | ||
| 
						 | 
					f17d74c8b7 | ||
| 
						 | 
					c88854c54c | ||
| 
						 | 
					f3779961d6 | ||
| 
						 | 
					d93fc29734 | ||
| 
						 | 
					c67918aca5 | ||
| 
						 | 
					a9f276c95a | ||
| 
						 | 
					7cee4894a5 | ||
| 
						 | 
					edf8bef813 | ||
| 
						 | 
					2081218398 | ||
| 
						 | 
					b100052453 | ||
| 
						 | 
					71636e895e | ||
| 
						 | 
					7ff9689b76 | ||
| 
						 | 
					5a4d819622 | ||
| 
						 | 
					3117d85648 | ||
| 
						 | 
					114133ecd2 | ||
| 
						 | 
					bf8a1197e4 | ||
| 
						 | 
					54c06a1fc0 | ||
| 
						 | 
					e77a42dfda | ||
| 
						 | 
					f83b4a2ba7 | ||
| 
						 | 
					d34e7b8d8a | ||
| 
						 | 
					fa0c7f3c66 | ||
| 
						 | 
					5f58645b41 | ||
| 
						 | 
					7ae0ec7573 | ||
| 
						 | 
					b1149cecaf | ||
| 
						 | 
					8f28d2be65 | ||
| 
						 | 
					d758b54ef8 | ||
| 
						 | 
					58293b4dc4 | ||
| 
						 | 
					f2083f4256 | ||
| 
						 | 
					6c7bd5804e | ||
| 
						 | 
					483ae21e89 | ||
| 
						 | 
					fc36d51e24 | ||
| 
						 | 
					f734565844 | ||
| 
						 | 
					8c718ba181 | ||
| 
						 | 
					8aaa2e7add | ||
| 
						 | 
					c8d8734601 | ||
| 
						 | 
					5c757e8255 | ||
| 
						 | 
					22f608f302 | ||
| 
						 | 
					9727405194 | 
@@ -1,6 +1,27 @@
 | 
			
		||||
FROM elixir:1.17-otp-27
 | 
			
		||||
 | 
			
		||||
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 mix local.hex --force
 | 
			
		||||
 
 | 
			
		||||
@@ -1,15 +1,30 @@
 | 
			
		||||
{
 | 
			
		||||
  "name": "wanderer-dev",
 | 
			
		||||
  "dockerComposeFile": ["./docker-compose.yml"],
 | 
			
		||||
  "extensions": ["jakebecker.elixir-ls"],
 | 
			
		||||
  "customizations": {
 | 
			
		||||
    "vscode": {
 | 
			
		||||
      "extensions": [
 | 
			
		||||
        "jakebecker.elixir-ls",
 | 
			
		||||
        "JakeBecker.elixir-ls",
 | 
			
		||||
        "dbaeumer.vscode-eslint",
 | 
			
		||||
        "esbenp.prettier-vscode"
 | 
			
		||||
      ],
 | 
			
		||||
      "settings": {
 | 
			
		||||
        "editor.formatOnSave": true,
 | 
			
		||||
        "search.exclude": {
 | 
			
		||||
          "**/doc": true
 | 
			
		||||
        },
 | 
			
		||||
        "elixirLS.dialyzerEnabled": false
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  },
 | 
			
		||||
  "service": "wanderer",
 | 
			
		||||
  "workspaceFolder": "/app",
 | 
			
		||||
  "shutdownAction": "stopCompose",
 | 
			
		||||
  "settings": {
 | 
			
		||||
    "editor.formatOnSave": true,
 | 
			
		||||
    "search.exclude": {
 | 
			
		||||
      "**/doc": true
 | 
			
		||||
    },
 | 
			
		||||
    "elixirLS.dialyzerEnabled": false
 | 
			
		||||
  }
 | 
			
		||||
  "features": {
 | 
			
		||||
    "ghcr.io/devcontainers/features/common-utils:2": {
 | 
			
		||||
      "networkArgs": ["--add-host=host.docker.internal:host-gateway"]
 | 
			
		||||
    }
 | 
			
		||||
  },
 | 
			
		||||
  "forwardPorts": [4444]
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -14,15 +14,15 @@ services:
 | 
			
		||||
 | 
			
		||||
  wanderer:
 | 
			
		||||
    environment:
 | 
			
		||||
      PORT: 8000
 | 
			
		||||
      PORT: 4444
 | 
			
		||||
      DB_HOST: db
 | 
			
		||||
      WEB_APP_URL: "http://localhost:8000"
 | 
			
		||||
      WEB_APP_URL: "http://localhost:4444"
 | 
			
		||||
      ERL_AFLAGS: "-kernel shell_history enabled"
 | 
			
		||||
    build:
 | 
			
		||||
      context: .
 | 
			
		||||
      dockerfile: Dockerfile
 | 
			
		||||
    ports:
 | 
			
		||||
      - 8000:8000
 | 
			
		||||
      - 4444:4444
 | 
			
		||||
    volumes:
 | 
			
		||||
      - ..:/app:delegated
 | 
			
		||||
      - ~/.gitconfig:/root/.gitconfig
 | 
			
		||||
 
 | 
			
		||||
@@ -7,3 +7,5 @@ export EVE_CLIENT_WITH_WALLET_SECRET="<EVE_CLIENT_WITH_WALLET_SECRET>"
 | 
			
		||||
export GIT_SHA="1111"
 | 
			
		||||
export WANDERER_INVITES="false"
 | 
			
		||||
export WANDERER_PUBLIC_API_DISABLED="false"
 | 
			
		||||
export WANDERER_CHARACTER_API_DISABLED="false"
 | 
			
		||||
export WANDERER_ZKILL_PRELOAD_DISABLED="false"
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										122
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										122
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							@@ -1,8 +1,6 @@
 | 
			
		||||
name: Build
 | 
			
		||||
 | 
			
		||||
on:
 | 
			
		||||
  pull_request:
 | 
			
		||||
    types: [opened, synchronize, reopened, labeled, unlabeled]
 | 
			
		||||
  push:
 | 
			
		||||
    branches:
 | 
			
		||||
      - main
 | 
			
		||||
@@ -41,8 +39,28 @@ jobs:
 | 
			
		||||
        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:
 | 
			
		||||
    name: 🛠 Build
 | 
			
		||||
    needs: manual-approval
 | 
			
		||||
    runs-on: ubuntu-22.04
 | 
			
		||||
    if: ${{ (github.ref == 'refs/heads/main') && github.event_name == 'push' }}
 | 
			
		||||
    permissions:
 | 
			
		||||
@@ -78,22 +96,23 @@ jobs:
 | 
			
		||||
          fetch-depth: 0
 | 
			
		||||
      - name: 😅 Cache deps
 | 
			
		||||
        id: cache-deps
 | 
			
		||||
        uses: actions/cache@v3
 | 
			
		||||
        uses: actions/cache@v4
 | 
			
		||||
        env:
 | 
			
		||||
          cache-name: cache-elixir-deps
 | 
			
		||||
        with:
 | 
			
		||||
          path: deps
 | 
			
		||||
          key: ${{ runner.os }}-mix-${{ env.cache-name }}-${{ hashFiles('**/mix.lock') }}
 | 
			
		||||
          path: |
 | 
			
		||||
            deps
 | 
			
		||||
          key: ${{ runner.os }}-mix-${{ matrix.elixir }}-${{ matrix.otp }}-${{ hashFiles('**/mix.lock') }}
 | 
			
		||||
          restore-keys: |
 | 
			
		||||
            ${{ runner.os }}-mix-${{ env.cache-name }}-
 | 
			
		||||
            ${{ runner.os }}-mix-${{ matrix.elixir }}-${{ matrix.otp }}-
 | 
			
		||||
      - name: 😅 Cache compiled build
 | 
			
		||||
        id: cache-build
 | 
			
		||||
        uses: actions/cache@v3
 | 
			
		||||
        uses: actions/cache@v4
 | 
			
		||||
        env:
 | 
			
		||||
          cache-name: cache-compiled-build
 | 
			
		||||
        with:
 | 
			
		||||
          path: |
 | 
			
		||||
            **/_build
 | 
			
		||||
            _build
 | 
			
		||||
          key: ${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-${{ hashFiles( '**/lib/**/*.{ex,eex}', '**/config/*.exs', '**/mix.exs' ) }}
 | 
			
		||||
          restore-keys: |
 | 
			
		||||
            ${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-
 | 
			
		||||
@@ -122,6 +141,9 @@ jobs:
 | 
			
		||||
    name: 🛠 Build Docker Images
 | 
			
		||||
    needs: build
 | 
			
		||||
    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
 | 
			
		||||
@@ -135,6 +157,7 @@ jobs:
 | 
			
		||||
      matrix:
 | 
			
		||||
        platform:
 | 
			
		||||
          - linux/amd64
 | 
			
		||||
          - linux/arm64
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Prepare
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -183,15 +206,28 @@ jobs:
 | 
			
		||||
          push: true
 | 
			
		||||
          context: .
 | 
			
		||||
          file: ./Dockerfile
 | 
			
		||||
          tags: ${{ env.REGISTRY_IMAGE }}:latest,${{ env.REGISTRY_IMAGE }}:${{ steps.get-latest-tag.outputs.tag }}
 | 
			
		||||
          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: Image digest
 | 
			
		||||
        run: echo ${{ steps.build.outputs.digest }}
 | 
			
		||||
      - 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
 | 
			
		||||
@@ -211,16 +247,54 @@ jobs:
 | 
			
		||||
          maxLength: 500
 | 
			
		||||
          truncationSymbol: "…"
 | 
			
		||||
 | 
			
		||||
      - name: Discord Webhook Action
 | 
			
		||||
        uses: tsickert/discord-webhook@v5.3.0
 | 
			
		||||
  merge:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
      - docker
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Download digests
 | 
			
		||||
        uses: actions/download-artifact@v4
 | 
			
		||||
        with:
 | 
			
		||||
          webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
 | 
			
		||||
          content: ${{ steps.get-content.outputs.string }}
 | 
			
		||||
          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 }}
 | 
			
		||||
 | 
			
		||||
  create-release:
 | 
			
		||||
    name: 🏷 Create Release
 | 
			
		||||
    runs-on: ubuntu-22.04
 | 
			
		||||
    needs: docker
 | 
			
		||||
    needs: [docker, merge]
 | 
			
		||||
    if: ${{ github.ref == 'refs/heads/main' && github.event_name == 'push' }}
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: ⬇️ Checkout repo
 | 
			
		||||
@@ -228,17 +302,11 @@ jobs:
 | 
			
		||||
        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: 🏷 Create Draft Release
 | 
			
		||||
        uses: softprops/action-gh-release@v1
 | 
			
		||||
        with:
 | 
			
		||||
          tag_name: ${{ steps.get-latest-tag.outputs.tag }}
 | 
			
		||||
          name: Release ${{ steps.get-latest-tag.outputs.tag }}
 | 
			
		||||
          tag_name: ${{ needs.docker.outputs.release-tag }}
 | 
			
		||||
          name: Release ${{ needs.docker.outputs.release-tag }}
 | 
			
		||||
          body: |
 | 
			
		||||
            ## Info
 | 
			
		||||
            Commit ${{ github.sha }} was deployed to `staging`. [See code diff](${{ github.event.compare }}).
 | 
			
		||||
@@ -248,3 +316,9 @@ jobs:
 | 
			
		||||
            ## How to Promote?
 | 
			
		||||
            In order to promote this to prod, edit the draft and press **"Publish release"**.
 | 
			
		||||
          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 }}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -26,6 +26,7 @@
 | 
			
		||||
 | 
			
		||||
.env
 | 
			
		||||
*.local.env
 | 
			
		||||
test/manual/.auto*
 | 
			
		||||
 | 
			
		||||
.direnv/
 | 
			
		||||
.cache/
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										1159
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										1159
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							@@ -9,6 +9,9 @@ WORKDIR /app
 | 
			
		||||
# set build ENV
 | 
			
		||||
ENV MIX_ENV="prod"
 | 
			
		||||
 | 
			
		||||
# Set ERL_FLAGS for ARM compatibility
 | 
			
		||||
ENV ERL_FLAGS="+JPperf true"
 | 
			
		||||
 | 
			
		||||
# install mix dependencies
 | 
			
		||||
COPY mix.exs mix.lock ./
 | 
			
		||||
RUN rm -Rf _build deps && mix deps.get --only $MIX_ENV
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										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))))
 | 
			
		||||
SHELL := /bin/bash
 | 
			
		||||
@@ -35,6 +35,11 @@ test t:
 | 
			
		||||
coverage cover co:
 | 
			
		||||
	mix test --cover
 | 
			
		||||
 | 
			
		||||
unit-tests ut:
 | 
			
		||||
	@echo "Running unit tests..."
 | 
			
		||||
	@find test/unit -name "*.exs" -exec elixir {} \;
 | 
			
		||||
	@echo "All unit tests completed."
 | 
			
		||||
 | 
			
		||||
versions v:
 | 
			
		||||
	@echo "Tool Versions"
 | 
			
		||||
	@cat .tool-versions
 | 
			
		||||
 
 | 
			
		||||
@@ -58,6 +58,7 @@ Now you can visit [`localhost:8000`](http://localhost:8000) from your browser.
 | 
			
		||||
- `root@0d0a785313b6:/app# apt update`
 | 
			
		||||
- `root@0d0a785313b6:/app# curl -sL https://deb.nodesource.com/setup_18.x  | bash -`
 | 
			
		||||
- `root@0d0a785313b6:/app# apt-get install nodejs inotify-tools -y`
 | 
			
		||||
- `root@0d0a785313b6:/app# npm install -g yarn`
 | 
			
		||||
- `root@0d0a785313b6:/app# mix setup`
 | 
			
		||||
 | 
			
		||||
- See how to run server in #Run section
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,3 @@
 | 
			
		||||
// import '@fontsource-variable/inter'
 | 
			
		||||
// import '@fontsource-variable/jetbrains-mono'
 | 
			
		||||
// import './lib/tailwind/index.css';
 | 
			
		||||
import './css/app.css';
 | 
			
		||||
 | 
			
		||||
import './lib/phoenix';
 | 
			
		||||
 
 | 
			
		||||
@@ -25,6 +25,10 @@ body {
 | 
			
		||||
  width: 400px; /* As IE6 ignores !important it will set width as 400px; */
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
body > div:first-of-type {
 | 
			
		||||
  min-height: 500px !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.lending-normal {
 | 
			
		||||
  font-family: 'Shentox', 'Rogan', sans-serif !important;
 | 
			
		||||
  font-weight: 500;
 | 
			
		||||
@@ -108,19 +112,19 @@ body {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.wd-characters-icons {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  transition:
 | 
			
		||||
    border-color 250ms,
 | 
			
		||||
    opacity 250ms;
 | 
			
		||||
  width: 35px;
 | 
			
		||||
  height: 35px;
 | 
			
		||||
  border-radius: 50%;
 | 
			
		||||
  border-width: 2px;
 | 
			
		||||
  border-style: solid;
 | 
			
		||||
  border-color: #5a5a5a;
 | 
			
		||||
  background-color: rgba(0, 0, 0, 0);
 | 
			
		||||
  cursor: pointer;
 | 
			
		||||
  opacity: 0.6;
 | 
			
		||||
  /*display: flex;*/
 | 
			
		||||
  /*transition:*/
 | 
			
		||||
  /*  border-color 250ms,*/
 | 
			
		||||
  /*  opacity 250ms;*/
 | 
			
		||||
  /*width: 35px;*/
 | 
			
		||||
  /*height: 35px;*/
 | 
			
		||||
  /*border-radius: 50%;*/
 | 
			
		||||
  /*border-width: 2px;*/
 | 
			
		||||
  /*border-style: solid;*/
 | 
			
		||||
  /*border-color: #5a5a5a;*/
 | 
			
		||||
  /*background-color: rgba(0, 0, 0, 0);*/
 | 
			
		||||
  /*cursor: pointer;*/
 | 
			
		||||
  /*opacity: 0.6;*/
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.wd-bg-default {
 | 
			
		||||
@@ -932,3 +936,66 @@ body {
 | 
			
		||||
  width: 16px;
 | 
			
		||||
  height: 16px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.verticalTabsContainer {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  min-height: 300px;
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview {
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: flex-start;
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-panels {
 | 
			
		||||
  padding: 6px 1rem !important;
 | 
			
		||||
  flex-grow: 1;
 | 
			
		||||
  height: 100%;
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-nav-container {
 | 
			
		||||
  border-right: none;
 | 
			
		||||
  height: 100%;
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-nav {
 | 
			
		||||
  flex-direction: column;
 | 
			
		||||
  width: 150px;
 | 
			
		||||
  min-height: 100%;
 | 
			
		||||
  border: none;
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-nav li {
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  border-right: 4px solid var(--surface-hover);
 | 
			
		||||
  background-color: var(--surface-card);
 | 
			
		||||
  transition:
 | 
			
		||||
    background-color 200ms,
 | 
			
		||||
    border-right-color 200ms;
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-nav li:hover {
 | 
			
		||||
  background-color: var(--surface-hover);
 | 
			
		||||
  border-right: 4px solid var(--surface-100);
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-nav li .p-tabview-nav-link {
 | 
			
		||||
  transition: color 200ms;
 | 
			
		||||
  justify-content: flex-end;
 | 
			
		||||
  padding: 10px;
 | 
			
		||||
  background-color: initial;
 | 
			
		||||
  border: none;
 | 
			
		||||
  color: var(--gray-400);
 | 
			
		||||
  border-radius: initial;
 | 
			
		||||
  font-weight: 400;
 | 
			
		||||
  margin: 0;
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-nav li.p-tabview-selected {
 | 
			
		||||
  background-color: var(--surface-50);
 | 
			
		||||
  border-right: 4px solid var(--primary-color);
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-nav li.p-tabview-selected .p-tabview-nav-link {
 | 
			
		||||
  font-weight: 600;
 | 
			
		||||
  color: var(--primary-color);
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-nav li.p-tabview-selected:hover {
 | 
			
		||||
  border-right: 4px solid var(--primary-color);
 | 
			
		||||
}
 | 
			
		||||
.verticalTabsContainer .p-tabview-panel {
 | 
			
		||||
  flex-grow: 1;
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,14 +1,14 @@
 | 
			
		||||
import { ErrorBoundary } from 'react-error-boundary';
 | 
			
		||||
import { PrimeReactProvider } from 'primereact/api';
 | 
			
		||||
import { ErrorBoundary } from 'react-error-boundary';
 | 
			
		||||
 | 
			
		||||
import { ReactFlowProvider } from 'reactflow';
 | 
			
		||||
import { MapHandlers } from '@/hooks/Mapper/types/mapHandlers.ts';
 | 
			
		||||
import { ErrorInfo, useCallback, useEffect, useRef } from 'react';
 | 
			
		||||
import { ReactFlowProvider } from 'reactflow';
 | 
			
		||||
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 { MapRootProvider } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
import './common-styles/main.scss';
 | 
			
		||||
 | 
			
		||||
const ErrorFallback = () => {
 | 
			
		||||
  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 { handleCommand, handleMapEvent, handleMapEvents } = useMapperHandlers(mapperHandlerRefs.current, hooksRef);
 | 
			
		||||
  const { handleCommand, handleMapEvent } = useMapperHandlers(mapperHandlerRefs.current, hooksRef);
 | 
			
		||||
 | 
			
		||||
  const logError = useCallback((error: Error, info: ErrorInfo) => {
 | 
			
		||||
    if (!hooksRef.current) {
 | 
			
		||||
@@ -35,7 +35,6 @@ export default function MapRoot({ hooks }) {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    hooksRef.current.handleEvent('map_event', handleMapEvent);
 | 
			
		||||
    hooksRef.current.handleEvent('map_events', handleMapEvents);
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
 
 | 
			
		||||
@@ -67,16 +67,22 @@
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.p-sortable-column {
 | 
			
		||||
  font-size: 12px;
 | 
			
		||||
  font-weight: bold;
 | 
			
		||||
  padding: 3px 4px;
 | 
			
		||||
.p-datatable-thead {
 | 
			
		||||
  th, th.p-sortable-column {
 | 
			
		||||
    font-size: 12px;
 | 
			
		||||
    font-weight: bold;
 | 
			
		||||
    padding: 3px 4px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.p-selectable-row td {
 | 
			
		||||
  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 {
 | 
			
		||||
  transform: scale(0.7);
 | 
			
		||||
 | 
			
		||||
@@ -93,6 +99,11 @@
 | 
			
		||||
.p-dropdown-item {
 | 
			
		||||
  padding: 0.25rem 0.5rem;
 | 
			
		||||
  font-size: 14px;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
 | 
			
		||||
  .p-dropdown-item-label {
 | 
			
		||||
    width: 100%;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.p-dropdown-item-group {
 | 
			
		||||
@@ -112,3 +123,78 @@
 | 
			
		||||
.p-autocomplete .p-autocomplete-multiple-container .p-autocomplete-token {
 | 
			
		||||
  margin-right: 0 !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Fixed sizes of Input switch */
 | 
			
		||||
.p-inputswitch {
 | 
			
		||||
  width: 2.0rem;
 | 
			
		||||
  height: 1.15rem;
 | 
			
		||||
 | 
			
		||||
  .p-inputswitch-slider:before {
 | 
			
		||||
    width: 0.8rem;
 | 
			
		||||
    height: 0.8rem;
 | 
			
		||||
    left: 0.14rem;
 | 
			
		||||
    margin-top: -0.385rem;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.p-highlight .p-inputswitch-slider:before {
 | 
			
		||||
    transform: translateX(0.8rem);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &:not(.p-disabled):has(.p-inputswitch-input:hover) .p-inputswitch-slider {
 | 
			
		||||
    background: rgb(255 255 255 / 21%);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.p-highlight .p-inputswitch-slider {
 | 
			
		||||
    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;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,11 +1,8 @@
 | 
			
		||||
/* Основной класс диалога */
 | 
			
		||||
body .p-dialog {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  flex-direction: column;
 | 
			
		||||
  //position: absolute;
 | 
			
		||||
  top: 0;
 | 
			
		||||
  left: 0;
 | 
			
		||||
  //visibility: hidden;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
  border-radius: 2px;
 | 
			
		||||
  box-shadow: 0 2px 10px 0 rgba(0,0,0,0.2);
 | 
			
		||||
@@ -29,12 +26,10 @@ body .p-dialog {
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Стиль видимого диалога */
 | 
			
		||||
.p-dialog-visible {
 | 
			
		||||
  visibility: visible;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Анимации */
 | 
			
		||||
.p-dialog-enter {
 | 
			
		||||
  opacity: 0;
 | 
			
		||||
}
 | 
			
		||||
@@ -53,31 +48,27 @@ body .p-dialog {
 | 
			
		||||
  transition: opacity 0.3s;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Заголовок диалога */
 | 
			
		||||
.p-dialog-header {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  justify-content: space-between;
 | 
			
		||||
  padding: 1rem;
 | 
			
		||||
  background: #f4f4f4;
 | 
			
		||||
  //border-bottom: 1px solid #ddd;
 | 
			
		||||
  height: 40px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Содержимое диалога */
 | 
			
		||||
.p-dialog-content {
 | 
			
		||||
  padding: 0.5rem;
 | 
			
		||||
  overflow-y: auto;
 | 
			
		||||
  flex: 1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Подвал диалога */
 | 
			
		||||
.p-dialog-footer {
 | 
			
		||||
  padding: 1rem;
 | 
			
		||||
  border-top: 1px solid #ddd;
 | 
			
		||||
  background: #f4f4f4;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Кнопка закрытия диалога */
 | 
			
		||||
.p-dialog-header-close {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
@@ -93,3 +84,12 @@ body .p-dialog {
 | 
			
		||||
.p-dialog-header-close .pi {
 | 
			
		||||
  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,77 @@
 | 
			
		||||
.vertical-tabs-container {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  min-height: 300px;
 | 
			
		||||
 | 
			
		||||
  .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);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .p-tabview-panel {
 | 
			
		||||
    flex-grow: 1;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,4 +1,6 @@
 | 
			
		||||
@import "fix-dialog";
 | 
			
		||||
@import "fix-popup";
 | 
			
		||||
@import "fix-tabs";
 | 
			
		||||
//@import "fix-input";
 | 
			
		||||
 | 
			
		||||
//@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);
 | 
			
		||||
}
 | 
			
		||||
@@ -4,10 +4,22 @@ 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 { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
import classes from './Characters.module.scss';
 | 
			
		||||
import { isDocked } from '@/hooks/Mapper/helpers/isDocked.ts';
 | 
			
		||||
import { PrimeIcons } from 'primereact/api';
 | 
			
		||||
 | 
			
		||||
const Characters = ({ data }: { data: CharacterTypeRaw[] }) => {
 | 
			
		||||
interface CharactersProps {
 | 
			
		||||
  data: CharacterTypeRaw[];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const Characters = ({ data }: CharactersProps) => {
 | 
			
		||||
  const [parent] = useAutoAnimate();
 | 
			
		||||
 | 
			
		||||
  const {
 | 
			
		||||
    data: { mainCharacterEveId, followingCharacterEveId },
 | 
			
		||||
  } = useMapRootState();
 | 
			
		||||
 | 
			
		||||
  const handleSelect = useCallback((character: CharacterTypeRaw) => {
 | 
			
		||||
    emitMapEvent({
 | 
			
		||||
      name: Commands.centerSystem,
 | 
			
		||||
@@ -21,21 +33,58 @@ const Characters = ({ data }: { data: CharacterTypeRaw[] }) => {
 | 
			
		||||
      className="flex flex-col items-center justify-center"
 | 
			
		||||
      onClick={() => handleSelect(character)}
 | 
			
		||||
    >
 | 
			
		||||
      <div className="tooltip tooltip-bottom" title={character.name}>
 | 
			
		||||
        <a
 | 
			
		||||
          className={clsx('wd-characters-icons wd-bg-default', { ['character-online']: character.online })}
 | 
			
		||||
      <div
 | 
			
		||||
        className={clsx(
 | 
			
		||||
          'overflow-hidden relative',
 | 
			
		||||
          'flex w-[35px] h-[35px] rounded-[4px] border-[1px] border-solid bg-transparent cursor-pointer',
 | 
			
		||||
          'transition-colors duration-250',
 | 
			
		||||
          {
 | 
			
		||||
            ['border-stone-800/90']: !character.online,
 | 
			
		||||
            ['border-lime-600/70']: character.online,
 | 
			
		||||
          },
 | 
			
		||||
        )}
 | 
			
		||||
        title={character.name}
 | 
			
		||||
      >
 | 
			
		||||
        {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)` }}
 | 
			
		||||
        ></a>
 | 
			
		||||
        ></div>
 | 
			
		||||
      </div>
 | 
			
		||||
    </li>
 | 
			
		||||
  ));
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <ul className="flex characters" id="characters" ref={parent}>
 | 
			
		||||
    <ul className="flex gap-1 characters" id="characters" ref={parent}>
 | 
			
		||||
      {items}
 | 
			
		||||
    </ul>
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// eslint-disable-next-line react/display-name
 | 
			
		||||
export default Characters;
 | 
			
		||||
 
 | 
			
		||||
@@ -6,6 +6,7 @@ import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/ty
 | 
			
		||||
 | 
			
		||||
export interface ContextMenuSystemProps {
 | 
			
		||||
  hubs: string[];
 | 
			
		||||
  userHubs: string[];
 | 
			
		||||
  contextMenuRef: RefObject<ContextMenu>;
 | 
			
		||||
  systemId: string | undefined;
 | 
			
		||||
  systems: SolarSystemRawType[];
 | 
			
		||||
@@ -13,6 +14,7 @@ export interface ContextMenuSystemProps {
 | 
			
		||||
  onLockToggle(): void;
 | 
			
		||||
  onOpenSettings(): void;
 | 
			
		||||
  onHubToggle(): void;
 | 
			
		||||
  onUserHubToggle(): void;
 | 
			
		||||
  onSystemTag(val?: string): void;
 | 
			
		||||
  onSystemStatus(val: number): void;
 | 
			
		||||
  onSystemLabels(val: string): void;
 | 
			
		||||
@@ -25,7 +27,7 @@ export const ContextMenuSystem: React.FC<ContextMenuSystemProps> = ({ contextMen
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <>
 | 
			
		||||
      <ContextMenu model={items} ref={contextMenuRef} breakpoint="767px" />
 | 
			
		||||
      <ContextMenu className="min-w-[200px]" model={items} ref={contextMenuRef} breakpoint="767px" />
 | 
			
		||||
    </>
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +0,0 @@
 | 
			
		||||
.active {
 | 
			
		||||
  background-color: rgba(98, 98, 98, 0.33);
 | 
			
		||||
}
 | 
			
		||||
@@ -1,3 +0,0 @@
 | 
			
		||||
.active {
 | 
			
		||||
  background-color: rgba(98, 98, 98, 0.33);
 | 
			
		||||
}
 | 
			
		||||
@@ -1 +1 @@
 | 
			
		||||
export * from './useTagMenu.ts';
 | 
			
		||||
export * from './useTagMenu.tsx';
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +0,0 @@
 | 
			
		||||
.active {
 | 
			
		||||
  background-color: rgba(98, 98, 98, 0.33);
 | 
			
		||||
}
 | 
			
		||||
@@ -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;
 | 
			
		||||
  }, []);
 | 
			
		||||
};
 | 
			
		||||
@@ -8,19 +8,25 @@ import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
 | 
			
		||||
 | 
			
		||||
interface UseContextMenuSystemHandlersProps {
 | 
			
		||||
  hubs: string[];
 | 
			
		||||
  userHubs: string[];
 | 
			
		||||
  systems: SolarSystemRawType[];
 | 
			
		||||
  outCommand: OutCommandHandler;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseContextMenuSystemHandlersProps) => {
 | 
			
		||||
export const useContextMenuSystemHandlers = ({
 | 
			
		||||
  systems,
 | 
			
		||||
  hubs,
 | 
			
		||||
  userHubs,
 | 
			
		||||
  outCommand,
 | 
			
		||||
}: UseContextMenuSystemHandlersProps) => {
 | 
			
		||||
  const contextMenuRef = useRef<ContextMenu | null>(null);
 | 
			
		||||
 | 
			
		||||
  const [system, setSystem] = useState<string>();
 | 
			
		||||
 | 
			
		||||
  const { deleteSystems } = useDeleteSystems();
 | 
			
		||||
 | 
			
		||||
  const ref = useRef({ hubs, system, systems, outCommand, deleteSystems });
 | 
			
		||||
  ref.current = { hubs, system, systems, outCommand, deleteSystems };
 | 
			
		||||
  const ref = useRef({ hubs, userHubs, system, systems, outCommand, deleteSystems });
 | 
			
		||||
  ref.current = { hubs, userHubs, system, systems, outCommand, deleteSystems };
 | 
			
		||||
 | 
			
		||||
  const open = useCallback((ev: any, systemId: string) => {
 | 
			
		||||
    setSystem(systemId);
 | 
			
		||||
@@ -72,6 +78,21 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
 | 
			
		||||
    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 onSystemTag = useCallback((tag?: string) => {
 | 
			
		||||
    const { system, outCommand } = ref.current;
 | 
			
		||||
    if (!system) {
 | 
			
		||||
@@ -104,7 +125,6 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
 | 
			
		||||
    setSystem(undefined);
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  const onSystemStatus = useCallback((status: number) => {
 | 
			
		||||
    const { system, outCommand } = ref.current;
 | 
			
		||||
    if (!system) {
 | 
			
		||||
@@ -177,6 +197,7 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
 | 
			
		||||
    onDeleteSystem,
 | 
			
		||||
    onLockToggle,
 | 
			
		||||
    onHubToggle,
 | 
			
		||||
    onUserHubToggle,
 | 
			
		||||
    onSystemTag,
 | 
			
		||||
    onSystemTemporaryName,
 | 
			
		||||
    onSystemStatus,
 | 
			
		||||
 
 | 
			
		||||
@@ -9,11 +9,14 @@ import { FastSystemActions } from '@/hooks/Mapper/components/contexts/components
 | 
			
		||||
import { useMapCheckPermissions } from '@/hooks/Mapper/mapRootProvider/hooks/api';
 | 
			
		||||
import { UserPermission } from '@/hooks/Mapper/types/permissions.ts';
 | 
			
		||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace.ts';
 | 
			
		||||
import { getSystemStaticInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
 | 
			
		||||
import { MapAddIcon, MapDeleteIcon, MapUserAddIcon, MapUserDeleteIcon } from '@/hooks/Mapper/icons';
 | 
			
		||||
 | 
			
		||||
export const useContextMenuSystemItems = ({
 | 
			
		||||
  onDeleteSystem,
 | 
			
		||||
  onLockToggle,
 | 
			
		||||
  onHubToggle,
 | 
			
		||||
  onUserHubToggle,
 | 
			
		||||
  onSystemTag,
 | 
			
		||||
  onSystemStatus,
 | 
			
		||||
  onSystemLabels,
 | 
			
		||||
@@ -22,6 +25,7 @@ export const useContextMenuSystemItems = ({
 | 
			
		||||
  onWaypointSet,
 | 
			
		||||
  systemId,
 | 
			
		||||
  hubs,
 | 
			
		||||
  userHubs,
 | 
			
		||||
  systems,
 | 
			
		||||
}: Omit<ContextMenuSystemProps, 'contextMenuRef'>) => {
 | 
			
		||||
  const getTags = useTagMenu(systems, systemId, onSystemTag);
 | 
			
		||||
@@ -32,6 +36,7 @@ export const useContextMenuSystemItems = ({
 | 
			
		||||
 | 
			
		||||
  return useMemo(() => {
 | 
			
		||||
    const system = systemId ? getSystemById(systems, systemId) : undefined;
 | 
			
		||||
    const systemStaticInfo = getSystemStaticInfo(systemId)!;
 | 
			
		||||
 | 
			
		||||
    if (!system || !systemId) {
 | 
			
		||||
      return [];
 | 
			
		||||
@@ -44,9 +49,9 @@ export const useContextMenuSystemItems = ({
 | 
			
		||||
          return (
 | 
			
		||||
            <FastSystemActions
 | 
			
		||||
              systemId={systemId}
 | 
			
		||||
              systemName={system.system_static_info.solar_system_name}
 | 
			
		||||
              regionName={system.system_static_info.region_name}
 | 
			
		||||
              isWH={isWormholeSpace(system.system_static_info.system_class)}
 | 
			
		||||
              systemName={systemStaticInfo.solar_system_name}
 | 
			
		||||
              regionName={systemStaticInfo.region_name}
 | 
			
		||||
              isWH={isWormholeSpace(systemStaticInfo.system_class)}
 | 
			
		||||
              showEdit
 | 
			
		||||
              onOpenSettings={onOpenSettings}
 | 
			
		||||
            />
 | 
			
		||||
@@ -57,12 +62,25 @@ export const useContextMenuSystemItems = ({
 | 
			
		||||
      getTags(),
 | 
			
		||||
      getStatus(),
 | 
			
		||||
      ...getLabels(),
 | 
			
		||||
      ...getWaypointMenu(systemId, system.system_static_info.system_class),
 | 
			
		||||
      ...getWaypointMenu(systemId, systemStaticInfo.system_class),
 | 
			
		||||
      {
 | 
			
		||||
        label: !hubs.includes(systemId) ? 'Add in Routes' : 'Remove from Routes',
 | 
			
		||||
        icon: PrimeIcons.MAP_MARKER,
 | 
			
		||||
        label: !hubs.includes(systemId) ? 'Add Route' : 'Remove Route',
 | 
			
		||||
        icon: !hubs.includes(systemId) ? (
 | 
			
		||||
          <MapAddIcon className="mr-1 relative left-[-2px]" />
 | 
			
		||||
        ) : (
 | 
			
		||||
          <MapDeleteIcon className="mr-1 relative left-[-2px]" />
 | 
			
		||||
        ),
 | 
			
		||||
        command: onHubToggle,
 | 
			
		||||
      },
 | 
			
		||||
      {
 | 
			
		||||
        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,
 | 
			
		||||
      },
 | 
			
		||||
      ...(system.locked
 | 
			
		||||
        ? canLockSystem
 | 
			
		||||
          ? [
 | 
			
		||||
 
 | 
			
		||||
@@ -11,6 +11,7 @@ import { FastSystemActions } from '@/hooks/Mapper/components/contexts/components
 | 
			
		||||
import { useJumpPlannerMenu } from '@/hooks/Mapper/components/contexts/hooks';
 | 
			
		||||
import { Route } from '@/hooks/Mapper/types/routes.ts';
 | 
			
		||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace.ts';
 | 
			
		||||
import { MapAddIcon, MapDeleteIcon } from '@/hooks/Mapper/icons';
 | 
			
		||||
 | 
			
		||||
export interface ContextMenuSystemInfoProps {
 | 
			
		||||
  systemStatics: Map<number, SolarSystemStaticInfoRaw>;
 | 
			
		||||
@@ -69,8 +70,12 @@ export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
 | 
			
		||||
      ...getJumpPlannerMenu(system, routes),
 | 
			
		||||
      ...getWaypointMenu(systemId, system.system_class),
 | 
			
		||||
      {
 | 
			
		||||
        label: !hubs.includes(systemId) ? 'Add in Routes' : 'Remove from Routes',
 | 
			
		||||
        icon: PrimeIcons.MAP_MARKER,
 | 
			
		||||
        label: !hubs.includes(systemId) ? 'Add Route' : 'Remove Route',
 | 
			
		||||
        icon: !hubs.includes(systemId) ? (
 | 
			
		||||
          <MapAddIcon className="mr-1 relative left-[-2px]" />
 | 
			
		||||
        ) : (
 | 
			
		||||
          <MapDeleteIcon className="mr-1 relative left-[-2px]" />
 | 
			
		||||
        ),
 | 
			
		||||
        command: onHubToggle,
 | 
			
		||||
      },
 | 
			
		||||
      ...(!systemOnMap
 | 
			
		||||
 
 | 
			
		||||
@@ -1,25 +1,25 @@
 | 
			
		||||
import * as React from 'react';
 | 
			
		||||
import { useCallback, useRef, useState } from 'react';
 | 
			
		||||
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 { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
 | 
			
		||||
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
 | 
			
		||||
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 {
 | 
			
		||||
  hubs: string[];
 | 
			
		||||
  outCommand: OutCommandHandler;
 | 
			
		||||
}
 | 
			
		||||
export const useContextMenuSystemInfoHandlers = () => {
 | 
			
		||||
  const { outCommand } = useMapRootState();
 | 
			
		||||
  const { hubs = [], toggleHubCommand } = useRouteProvider();
 | 
			
		||||
 | 
			
		||||
export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand }: UseContextMenuSystemHandlersProps) => {
 | 
			
		||||
  const contextMenuRef = useRef<ContextMenu | null>(null);
 | 
			
		||||
 | 
			
		||||
  const [system, setSystem] = useState<string>();
 | 
			
		||||
  const routeRef = useRef<(SolarSystemStaticInfoRaw | undefined)[]>([]);
 | 
			
		||||
 | 
			
		||||
  const ref = useRef({ hubs, system, outCommand });
 | 
			
		||||
  ref.current = { hubs, system, outCommand };
 | 
			
		||||
  const ref = useRef({ hubs, system, outCommand, toggleHubCommand });
 | 
			
		||||
  ref.current = { hubs, system, outCommand, toggleHubCommand };
 | 
			
		||||
 | 
			
		||||
  const open = useCallback(
 | 
			
		||||
    (ev: React.SyntheticEvent, systemId: string, route: (SolarSystemStaticInfoRaw | undefined)[]) => {
 | 
			
		||||
@@ -33,17 +33,12 @@ export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand }: UseContex
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const onHubToggle = useCallback(() => {
 | 
			
		||||
    const { hubs, system, outCommand } = ref.current;
 | 
			
		||||
    const { system } = ref.current;
 | 
			
		||||
    if (!system) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    outCommand({
 | 
			
		||||
      type: !hubs.includes(system) ? OutCommand.addHub : OutCommand.deleteHub,
 | 
			
		||||
      data: {
 | 
			
		||||
        system_id: system,
 | 
			
		||||
      },
 | 
			
		||||
    });
 | 
			
		||||
    ref.current.toggleHubCommand(system);
 | 
			
		||||
    setSystem(undefined);
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
@@ -59,6 +54,8 @@ export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand }: UseContex
 | 
			
		||||
        system_id: solarSystemId,
 | 
			
		||||
      },
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    // TODO add it to some queue
 | 
			
		||||
    setTimeout(() => {
 | 
			
		||||
      emitMapEvent({
 | 
			
		||||
        name: Commands.centerSystem,
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
import { useCallback, useRef } from 'react';
 | 
			
		||||
import { LayoutEventBlocker, 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 clsx from 'clsx';
 | 
			
		||||
 
 | 
			
		||||
@@ -4,8 +4,8 @@ import { useCallback } from 'react';
 | 
			
		||||
import { isPossibleSpace } from '@/hooks/Mapper/components/map/helpers/isKnownSpace.ts';
 | 
			
		||||
import { Route } from '@/hooks/Mapper/types/routes.ts';
 | 
			
		||||
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 { 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 criminalSpace = [SOLAR_SYSTEM_CLASS_IDS.ls, SOLAR_SYSTEM_CLASS_IDS.ns];
 | 
			
		||||
@@ -47,7 +47,7 @@ export const useJumpPlannerMenu = (
 | 
			
		||||
        return [];
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const origin = getSystemById(systems, systemIdFrom)?.system_static_info;
 | 
			
		||||
      const origin = getSystemStaticInfo(systemIdFrom);
 | 
			
		||||
 | 
			
		||||
      if (!origin) {
 | 
			
		||||
        return [];
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +0,0 @@
 | 
			
		||||
.active {
 | 
			
		||||
  background-color: rgba(98, 98, 98, 0.33);
 | 
			
		||||
}
 | 
			
		||||
@@ -1,2 +1,3 @@
 | 
			
		||||
export * from './useSystemInfo';
 | 
			
		||||
export * from './useGetOwnOnlineCharacters';
 | 
			
		||||
export * from './useElementWidth';
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										43
									
								
								assets/js/hooks/Mapper/components/hooks/useElementWidth.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								assets/js/hooks/Mapper/components/hooks/useElementWidth.ts
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,43 @@
 | 
			
		||||
import { useState, useLayoutEffect, RefObject } from 'react';
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * useElementWidth
 | 
			
		||||
 *
 | 
			
		||||
 * A custom hook that accepts a ref to an HTML element and returns its current width.
 | 
			
		||||
 * It uses a ResizeObserver and window resize listener to update the width when necessary.
 | 
			
		||||
 *
 | 
			
		||||
 * @param ref - A RefObject pointing to an HTML element.
 | 
			
		||||
 * @returns The current width of the element.
 | 
			
		||||
 */
 | 
			
		||||
export function useElementWidth<T extends HTMLElement>(ref: RefObject<T>): number {
 | 
			
		||||
  const [width, setWidth] = useState<number>(0);
 | 
			
		||||
 | 
			
		||||
  useLayoutEffect(() => {
 | 
			
		||||
    const updateWidth = () => {
 | 
			
		||||
      if (ref.current) {
 | 
			
		||||
        const newWidth = ref.current.getBoundingClientRect().width;
 | 
			
		||||
        if (newWidth > 0) {
 | 
			
		||||
          setWidth(newWidth);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    updateWidth(); // Initial measurement
 | 
			
		||||
 | 
			
		||||
    const observer = new ResizeObserver(() => {
 | 
			
		||||
      const id = setTimeout(updateWidth, 100);
 | 
			
		||||
      return () => clearTimeout(id);
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    if (ref.current) {
 | 
			
		||||
      observer.observe(ref.current);
 | 
			
		||||
    }
 | 
			
		||||
    window.addEventListener("resize", updateWidth);
 | 
			
		||||
    return () => {
 | 
			
		||||
      observer.disconnect();
 | 
			
		||||
      window.removeEventListener("resize", updateWidth);
 | 
			
		||||
    };
 | 
			
		||||
  }, [ref]);
 | 
			
		||||
 | 
			
		||||
  return width;
 | 
			
		||||
}
 | 
			
		||||
@@ -1,7 +1,7 @@
 | 
			
		||||
import { getSystemById } from '@/hooks/Mapper/helpers';
 | 
			
		||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
import { useMemo } from 'react';
 | 
			
		||||
import { getSystemById } from '@/hooks/Mapper/helpers';
 | 
			
		||||
import { useLoadSystemStatic } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic.ts';
 | 
			
		||||
import { getSystemStaticInfo } from '../../mapRootProvider/hooks/useLoadSystemStatic';
 | 
			
		||||
 | 
			
		||||
interface UseSystemInfoProps {
 | 
			
		||||
  systemId: string;
 | 
			
		||||
@@ -12,14 +12,12 @@ export const useSystemInfo = ({ systemId }: UseSystemInfoProps) => {
 | 
			
		||||
    data: { systems, connections },
 | 
			
		||||
  } = useMapRootState();
 | 
			
		||||
 | 
			
		||||
  const { systems: systemStatics } = useLoadSystemStatic({ systems: [systemId] });
 | 
			
		||||
 | 
			
		||||
  return useMemo(() => {
 | 
			
		||||
    const staticInfo = systemStatics.get(parseInt(systemId));
 | 
			
		||||
    const staticInfo = getSystemStaticInfo(parseInt(systemId));
 | 
			
		||||
    const dynamicInfo = getSystemById(systems, systemId);
 | 
			
		||||
 | 
			
		||||
    if (!staticInfo || !dynamicInfo) {
 | 
			
		||||
      throw new Error(`Error on getting system ${systemId}`);
 | 
			
		||||
      return { dynamicInfo, staticInfo, leadsTo: [] };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const leadsTo = connections
 | 
			
		||||
@@ -29,5 +27,5 @@ export const useSystemInfo = ({ systemId }: UseSystemInfoProps) => {
 | 
			
		||||
      .filter(x => x !== systemId);
 | 
			
		||||
 | 
			
		||||
    return { dynamicInfo, staticInfo, leadsTo };
 | 
			
		||||
  }, [systemStatics, systemId, systems, connections]);
 | 
			
		||||
  }, [systemId, systems, connections]);
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -2,10 +2,10 @@
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  height: 100%;
 | 
			
		||||
 | 
			
		||||
  background-color: var(--rf-bg-color, #000000);
 | 
			
		||||
  background-color: var(--rf-bg-color, #0C0A09);
 | 
			
		||||
 | 
			
		||||
  &.BackgroundAlternateColor {
 | 
			
		||||
    background-color: var(--rf-soft-bg-color, #171717);
 | 
			
		||||
    --rf-node-bg-color: var(--rf-node-soft-bg-color, #2b2b2b);
 | 
			
		||||
    --rf-node-bg-color: var(--rf-node-soft-bg-color, #202020);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,6 @@
 | 
			
		||||
import { ForwardedRef, forwardRef, MouseEvent, useCallback, useEffect } from 'react';
 | 
			
		||||
import { ForwardedRef, forwardRef, MouseEvent, useCallback, useEffect, useMemo } from 'react';
 | 
			
		||||
import ReactFlow, {
 | 
			
		||||
  Background,
 | 
			
		||||
  ConnectionMode,
 | 
			
		||||
  Edge,
 | 
			
		||||
  MiniMap,
 | 
			
		||||
  Node,
 | 
			
		||||
@@ -17,16 +16,16 @@ import ReactFlow, {
 | 
			
		||||
import 'reactflow/dist/style.css';
 | 
			
		||||
import classes from './Map.module.scss';
 | 
			
		||||
import { MapProvider, useMapState } from './MapProvider';
 | 
			
		||||
import { useNodesState, useEdgesState, useMapHandlers, useUpdateNodes } from './hooks';
 | 
			
		||||
import { useEdgesState, useMapHandlers, useNodesState, useUpdateNodes } from './hooks';
 | 
			
		||||
import { MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
 | 
			
		||||
import {
 | 
			
		||||
  ContextMenuConnection,
 | 
			
		||||
  ContextMenuRoot,
 | 
			
		||||
  SolarSystemEdge,
 | 
			
		||||
  SolarSystemNode,
 | 
			
		||||
  useContextMenuConnectionHandlers,
 | 
			
		||||
  useContextMenuRootHandlers,
 | 
			
		||||
} from './components';
 | 
			
		||||
import { getBehaviorForTheme } from './helpers/getThemeBehavior';
 | 
			
		||||
import { OnMapAddSystemCallback, OnMapSelectionChange } from './map.types';
 | 
			
		||||
import { SESSION_KEY } from '@/hooks/Mapper/constants.ts';
 | 
			
		||||
import { SolarSystemConnection, SolarSystemRawType } from '@/hooks/Mapper/types';
 | 
			
		||||
@@ -75,21 +74,16 @@ const initialEdges = [
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const nodeTypes = {
 | 
			
		||||
  // eslint-disable-next-line @typescript-eslint/ban-ts-comment
 | 
			
		||||
  // @ts-ignore
 | 
			
		||||
  custom: SolarSystemNode,
 | 
			
		||||
} as never;
 | 
			
		||||
 | 
			
		||||
const edgeTypes = {
 | 
			
		||||
  floating: SolarSystemEdge,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const MAP_ROOT_ID = 'MAP_ROOT_ID';
 | 
			
		||||
 | 
			
		||||
interface MapCompProps {
 | 
			
		||||
  refn: ForwardedRef<MapHandlers>;
 | 
			
		||||
  onCommand: OutCommandHandler;
 | 
			
		||||
  onSelectionChange: OnMapSelectionChange;
 | 
			
		||||
  onManualDelete(systems: string[]): void;
 | 
			
		||||
  onConnectionInfoClick?(e: SolarSystemConnection): void;
 | 
			
		||||
  onAddSystem?: OnMapAddSystemCallback;
 | 
			
		||||
  onSelectionContextMenu?: NodeSelectionMouseHandler;
 | 
			
		||||
@@ -111,7 +105,6 @@ const MapComp = ({
 | 
			
		||||
  onSystemContextMenu,
 | 
			
		||||
  onConnectionInfoClick,
 | 
			
		||||
  onSelectionContextMenu,
 | 
			
		||||
  onManualDelete,
 | 
			
		||||
  isShowMinimap,
 | 
			
		||||
  showKSpaceBG,
 | 
			
		||||
  isThickConnections,
 | 
			
		||||
@@ -120,7 +113,7 @@ const MapComp = ({
 | 
			
		||||
  theme,
 | 
			
		||||
  onAddSystem,
 | 
			
		||||
}: MapCompProps) => {
 | 
			
		||||
  const { getNode, getNodes } = useReactFlow();
 | 
			
		||||
  const { getNodes } = useReactFlow();
 | 
			
		||||
  const [nodes, , onNodesChange] = useNodesState<Node<SolarSystemRawType>>(initialNodes);
 | 
			
		||||
  const [edges, , onEdgesChange] = useEdgesState<Edge<SolarSystemConnection>>(initialEdges);
 | 
			
		||||
 | 
			
		||||
@@ -130,6 +123,13 @@ const MapComp = ({
 | 
			
		||||
  const { handleConnectionContext, ...connectionCtxProps } = useContextMenuConnectionHandlers();
 | 
			
		||||
  const { update } = useMapState();
 | 
			
		||||
  const { variant, gap, size, color } = useBackgroundVars(theme);
 | 
			
		||||
  const { isPanAndDrag, nodeComponent, connectionMode } = getBehaviorForTheme(theme || 'default');
 | 
			
		||||
 | 
			
		||||
  const nodeTypes = useMemo(() => {
 | 
			
		||||
    return {
 | 
			
		||||
      custom: nodeComponent,
 | 
			
		||||
    };
 | 
			
		||||
  }, [nodeComponent]);
 | 
			
		||||
 | 
			
		||||
  const onConnect: OnConnect = useCallback(
 | 
			
		||||
    params => {
 | 
			
		||||
@@ -186,8 +186,6 @@ const MapComp = ({
 | 
			
		||||
 | 
			
		||||
  const handleNodesChange = useCallback(
 | 
			
		||||
    (changes: NodeChange[]) => {
 | 
			
		||||
      const systemsIdsToRemove: string[] = [];
 | 
			
		||||
 | 
			
		||||
      // prevents single node deselection on background / same node click
 | 
			
		||||
      // allows deseletion of all nodes if multiple are currently selected
 | 
			
		||||
      if (changes.length === 1 && changes[0].type == 'select' && changes[0].selected === false) {
 | 
			
		||||
@@ -195,30 +193,12 @@ const MapComp = ({
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      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];
 | 
			
		||||
      }, [] as NodeChange[]);
 | 
			
		||||
 | 
			
		||||
      if (systemsIdsToRemove.length > 0) {
 | 
			
		||||
        onManualDelete(systemsIdsToRemove);
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      onNodesChange(nextChanges);
 | 
			
		||||
    },
 | 
			
		||||
    [getNode, onManualDelete, onNodesChange],
 | 
			
		||||
    [getNodes, onNodesChange],
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
@@ -231,7 +211,10 @@ const MapComp = ({
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <>
 | 
			
		||||
      <div className={clsx(classes.MapRoot, { [classes.BackgroundAlternateColor]: isSoftBackground })}>
 | 
			
		||||
      <div
 | 
			
		||||
        data-window-id={MAP_ROOT_ID}
 | 
			
		||||
        className={clsx(classes.MapRoot, { [classes.BackgroundAlternateColor]: isSoftBackground })}
 | 
			
		||||
      >
 | 
			
		||||
        <ReactFlow
 | 
			
		||||
          nodes={nodes}
 | 
			
		||||
          edges={edges}
 | 
			
		||||
@@ -243,7 +226,7 @@ const MapComp = ({
 | 
			
		||||
          defaultViewport={getViewPortFromStore()}
 | 
			
		||||
          edgeTypes={edgeTypes}
 | 
			
		||||
          nodeTypes={nodeTypes}
 | 
			
		||||
          connectionMode={ConnectionMode.Loose}
 | 
			
		||||
          connectionMode={connectionMode}
 | 
			
		||||
          snapToGrid
 | 
			
		||||
          nodeDragThreshold={10}
 | 
			
		||||
          onNodeDragStop={handleDragStop}
 | 
			
		||||
@@ -251,6 +234,10 @@ const MapComp = ({
 | 
			
		||||
          onConnectStart={() => update({ isConnecting: true })}
 | 
			
		||||
          onConnectEnd={() => update({ isConnecting: false })}
 | 
			
		||||
          onNodeMouseEnter={(_, node) => update({ hoverNodeId: node.id })}
 | 
			
		||||
          onPaneClick={event => {
 | 
			
		||||
            event.preventDefault();
 | 
			
		||||
            event.stopPropagation();
 | 
			
		||||
          }}
 | 
			
		||||
          // onKeyUp=
 | 
			
		||||
          onNodeMouseLeave={() => update({ hoverNodeId: null })}
 | 
			
		||||
          onEdgeClick={(_, t) => {
 | 
			
		||||
@@ -271,7 +258,13 @@ const MapComp = ({
 | 
			
		||||
          minZoom={0.2}
 | 
			
		||||
          maxZoom={1.5}
 | 
			
		||||
          elevateNodesOnSelect
 | 
			
		||||
          deleteKeyCode={['Delete']}
 | 
			
		||||
          deleteKeyCode={['']}
 | 
			
		||||
          {...(isPanAndDrag
 | 
			
		||||
            ? {
 | 
			
		||||
                selectionOnDrag: true,
 | 
			
		||||
                panOnDrag: [2],
 | 
			
		||||
              }
 | 
			
		||||
            : {})}
 | 
			
		||||
          // TODO need create clear example with problem with that flag
 | 
			
		||||
          //  if system is not visible edge not drawing (and any render in Custom node is not happening)
 | 
			
		||||
          // onlyRenderVisibleElements
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
import React, { createContext, useContext } from 'react';
 | 
			
		||||
import { OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
 | 
			
		||||
import { MapUnionTypes } from '@/hooks/Mapper/types';
 | 
			
		||||
import { MapUnionTypes, SystemSignature } from '@/hooks/Mapper/types';
 | 
			
		||||
import { ContextStoreDataUpdate, useContextStore } from '@/hooks/Mapper/utils';
 | 
			
		||||
 | 
			
		||||
export type MapData = MapUnionTypes & {
 | 
			
		||||
@@ -9,6 +9,7 @@ export type MapData = MapUnionTypes & {
 | 
			
		||||
  visibleNodes: Set<string>;
 | 
			
		||||
  showKSpaceBG: boolean;
 | 
			
		||||
  isThickConnections: boolean;
 | 
			
		||||
  linkedSigEveId: string;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
interface MapProviderProps {
 | 
			
		||||
@@ -29,10 +30,16 @@ const INITIAL_DATA: MapData = {
 | 
			
		||||
  isConnecting: false,
 | 
			
		||||
  connections: [],
 | 
			
		||||
  hoverNodeId: null,
 | 
			
		||||
  linkedSigEveId: '',
 | 
			
		||||
  visibleNodes: new Set(),
 | 
			
		||||
  showKSpaceBG: false,
 | 
			
		||||
  isThickConnections: false,
 | 
			
		||||
  userPermissions: {},
 | 
			
		||||
  systemSignatures: {} as Record<string, SystemSignature[]>,
 | 
			
		||||
  options: {} as Record<string, string | boolean>,
 | 
			
		||||
  isSubscriptionActive: false,
 | 
			
		||||
  mainCharacterEveId: null,
 | 
			
		||||
  followingCharacterEveId: null,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export interface MapContextProps {
 | 
			
		||||
@@ -48,7 +55,7 @@ const MapContext = createContext<MapContextProps>({
 | 
			
		||||
  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 });
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
 
 | 
			
		||||
@@ -2,7 +2,6 @@ import React, { RefObject, useMemo } from 'react';
 | 
			
		||||
import { ContextMenu } from 'primereact/contextmenu';
 | 
			
		||||
import { PrimeIcons } from 'primereact/api';
 | 
			
		||||
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 clsx from 'clsx';
 | 
			
		||||
import classes from './ContextMenuConnection.module.scss';
 | 
			
		||||
@@ -14,6 +13,7 @@ import {
 | 
			
		||||
  SHIP_SIZES_NAMES_SHORT,
 | 
			
		||||
  SHIP_SIZES_SIZE,
 | 
			
		||||
} from '@/hooks/Mapper/components/map/constants.ts';
 | 
			
		||||
import { Edge } from 'reactflow';
 | 
			
		||||
 | 
			
		||||
export interface ContextMenuConnectionProps {
 | 
			
		||||
  contextMenuRef: RefObject<ContextMenu>;
 | 
			
		||||
 
 | 
			
		||||
@@ -1,9 +1,8 @@
 | 
			
		||||
import { EdgeMouseHandler } from 'reactflow';
 | 
			
		||||
import { Edge, EdgeMouseHandler } from 'reactflow';
 | 
			
		||||
import { useCallback, useRef, useState } from 'react';
 | 
			
		||||
import { ContextMenu } from 'primereact/contextmenu';
 | 
			
		||||
import { useMapState } from '../../MapProvider.tsx';
 | 
			
		||||
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 { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,48 @@
 | 
			
		||||
.KillsBookmark {
 | 
			
		||||
    display: inline-flex;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    font-size: 8px;
 | 
			
		||||
    font-weight: 700;
 | 
			
		||||
    border: 0;
 | 
			
		||||
    border-radius: 5px 5px 0 0;
 | 
			
		||||
    padding: 4px 3px;
 | 
			
		||||
 | 
			
		||||
  }
 | 
			
		||||
  
 | 
			
		||||
.KillsBookmarkWithIcon {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    margin-top: -2px;
 | 
			
		||||
    text-shadow: 0 0 3px #000;
 | 
			
		||||
    padding-right: 2px;
 | 
			
		||||
    height: 8px;
 | 
			
		||||
    font-size: 8px;
 | 
			
		||||
    line-height: 12px;
 | 
			
		||||
    font-weight: 700;
 | 
			
		||||
    text-size-adjust: 100%;
 | 
			
		||||
 | 
			
		||||
    .pi {
 | 
			
		||||
        font-size: 9px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .text {
 | 
			
		||||
        font-size: 9px;
 | 
			
		||||
        font-family: var(--rf-node-font-family, inherit) !important;
 | 
			
		||||
        font-weight: var(--rf-node-font-weight, inherit) !important;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.TooltipContainer {
 | 
			
		||||
    background-color: #1a1a1a;
 | 
			
		||||
    color: #fff;
 | 
			
		||||
    padding: 3px;
 | 
			
		||||
    box-shadow: 0 2px 8px rgba(0, 0, 0, 0.3);
 | 
			
		||||
    border-radius: 2px;
 | 
			
		||||
    pointer-events: auto;
 | 
			
		||||
    max-width: 500px;
 | 
			
		||||
    max-height: 300px;
 | 
			
		||||
    overflow-y: auto;
 | 
			
		||||
    overflow-x: hidden;
 | 
			
		||||
  }
 | 
			
		||||
@@ -0,0 +1,64 @@
 | 
			
		||||
import { useMemo } from 'react';
 | 
			
		||||
import { useKillsCounter } from '../../hooks/useKillsCounter';
 | 
			
		||||
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
 | 
			
		||||
import { WithChildren, WithClassName } from '@/hooks/Mapper/types/common';
 | 
			
		||||
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 />
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
      }
 | 
			
		||||
      className={className}
 | 
			
		||||
      tooltipClassName="!px-0"
 | 
			
		||||
      size={size}
 | 
			
		||||
      interactive={true}
 | 
			
		||||
    >
 | 
			
		||||
      {children}
 | 
			
		||||
    </WdTooltipWrapper>
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
@@ -0,0 +1,54 @@
 | 
			
		||||
.TooltipActive {
 | 
			
		||||
  pointer-events: auto !important;
 | 
			
		||||
  position: relative;
 | 
			
		||||
  z-index: 3;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.hoverTarget {
 | 
			
		||||
  padding: 0.5rem;
 | 
			
		||||
  margin: -0.5rem;
 | 
			
		||||
  display: inline-block;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.localCounter {
 | 
			
		||||
  mix-blend-mode: screen;
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  gap: 1px;
 | 
			
		||||
  position: relative;
 | 
			
		||||
  top: 1px;
 | 
			
		||||
  color: var(--rf-node-local-counter);
 | 
			
		||||
 | 
			
		||||
  &.hasUserCharacters {
 | 
			
		||||
    color: var(--rf-has-user-characters);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  & > i {
 | 
			
		||||
    font-size: 9px;
 | 
			
		||||
    position: relative;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  & > span {
 | 
			
		||||
    font-size: 9px;
 | 
			
		||||
    line-height: 9px;
 | 
			
		||||
    font-weight: var(--rf-local-counter-font-weight, 500);
 | 
			
		||||
 | 
			
		||||
    @-moz-document url-prefix() {
 | 
			
		||||
      position: relative;
 | 
			
		||||
      top: -1px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.Pathfinder {
 | 
			
		||||
  .localCounter {
 | 
			
		||||
    @-moz-document url-prefix() {
 | 
			
		||||
      top: 0;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    & > span {
 | 
			
		||||
      position: relative;
 | 
			
		||||
      top: -1px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,61 @@
 | 
			
		||||
import { useMemo } from 'react';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
 | 
			
		||||
import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit/WdTooltip';
 | 
			
		||||
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 { useTheme } from '@/hooks/Mapper/hooks/useTheme.ts';
 | 
			
		||||
import { AvailableThemes } from '@/hooks/Mapper/mapRootProvider/types.ts';
 | 
			
		||||
 | 
			
		||||
interface LocalCounterProps {
 | 
			
		||||
  localCounterCharacters: Array<CharItemProps>;
 | 
			
		||||
  hasUserCharacters: boolean;
 | 
			
		||||
  showIcon?: boolean;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const LocalCounter = ({ localCounterCharacters, hasUserCharacters, showIcon = true }: LocalCounterProps) => {
 | 
			
		||||
  const [settings] = useLocalCharacterWidgetSettings();
 | 
			
		||||
  const itemTemplate = useLocalCharactersItemTemplate(settings.showShipName);
 | 
			
		||||
  const theme = useTheme();
 | 
			
		||||
 | 
			
		||||
  const pilotTooltipContent = useMemo(() => {
 | 
			
		||||
    return (
 | 
			
		||||
      <div
 | 
			
		||||
        style={{
 | 
			
		||||
          width: '100%',
 | 
			
		||||
          minWidth: '300px',
 | 
			
		||||
          overflow: 'hidden',
 | 
			
		||||
        }}
 | 
			
		||||
      >
 | 
			
		||||
        <LocalCharactersList items={localCounterCharacters} itemTemplate={itemTemplate} itemSize={26} autoSize={true} />
 | 
			
		||||
      </div>
 | 
			
		||||
    );
 | 
			
		||||
  }, [localCounterCharacters, itemTemplate]);
 | 
			
		||||
 | 
			
		||||
  if (localCounterCharacters.length === 0) {
 | 
			
		||||
    return null;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      className={clsx(classes.TooltipActive, {
 | 
			
		||||
        [classes.Pathfinder]: theme === AvailableThemes.pathfinder,
 | 
			
		||||
      })}
 | 
			
		||||
    >
 | 
			
		||||
      <WdTooltipWrapper content={pilotTooltipContent} position={TooltipPosition.right} offset={0} interactive={true}>
 | 
			
		||||
        <div className={clsx(classes.hoverTarget)}>
 | 
			
		||||
          <div
 | 
			
		||||
            className={clsx(classes.localCounter, {
 | 
			
		||||
              [classes.hasUserCharacters]: hasUserCharacters,
 | 
			
		||||
            })}
 | 
			
		||||
          >
 | 
			
		||||
            {showIcon && <i className="pi pi-users" />}
 | 
			
		||||
            <span>{localCounterCharacters.length}</span>
 | 
			
		||||
          </div>
 | 
			
		||||
        </div>
 | 
			
		||||
      </WdTooltipWrapper>
 | 
			
		||||
    </div>
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
@@ -1,278 +0,0 @@
 | 
			
		||||
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
 | 
			
		||||
 | 
			
		||||
.RootCustomNode {
 | 
			
		||||
  background-color: var(--rf-node-bg-color, #202020) !important;
 | 
			
		||||
  position: relative;
 | 
			
		||||
  z-index: 1;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
  font-weight: var(--rf-node-font-weight, bold);
 | 
			
		||||
  color: var(--rf-text-color, #ffffff);
 | 
			
		||||
  font-family: var(--rf-node-font, inherit);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Region backgrounds for optional tinted headers, etc. */
 | 
			
		||||
.Mataria,
 | 
			
		||||
.Amarria,
 | 
			
		||||
.Gallente,
 | 
			
		||||
.Caldaria {
 | 
			
		||||
  &::before {
 | 
			
		||||
    content: '';
 | 
			
		||||
    position: absolute;
 | 
			
		||||
    inset: 0;
 | 
			
		||||
    background-size: cover;
 | 
			
		||||
    background-position: 50% 50%;
 | 
			
		||||
    background-repeat: no-repeat;
 | 
			
		||||
    z-index: -1;
 | 
			
		||||
    border-radius: 3px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
.Mataria::before {
 | 
			
		||||
  background-image: url('/images/mataria-180.png');
 | 
			
		||||
  opacity: 0.6;
 | 
			
		||||
  background-position-x: 1px;
 | 
			
		||||
  background-position-y: -14px;
 | 
			
		||||
}
 | 
			
		||||
.Caldaria::before {
 | 
			
		||||
  background-image: url('/images/caldaria-180.png');
 | 
			
		||||
  opacity: 0.6;
 | 
			
		||||
  background-position-x: 1px;
 | 
			
		||||
  background-position-y: -10px;
 | 
			
		||||
}
 | 
			
		||||
.Amarria::before {
 | 
			
		||||
  opacity: 0.45;
 | 
			
		||||
  background-image: url('/images/amarr-180.png');
 | 
			
		||||
  background-position-x: 0;
 | 
			
		||||
  background-position-y: -13px;
 | 
			
		||||
}
 | 
			
		||||
.Gallente::before {
 | 
			
		||||
  opacity: 0.5;
 | 
			
		||||
  background-image: url('/images/gallente-180.png');
 | 
			
		||||
  background-position-x: 1px;
 | 
			
		||||
  background-position-y: 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.selected {
 | 
			
		||||
  border-color: var(--pastel-pink, #d291bc);
 | 
			
		||||
  box-shadow: 0 0 10px #9a1af1c2;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.eve-system-status-home {
 | 
			
		||||
  border: 1px solid var(--eve-solar-system-status-color-home-dark30);
 | 
			
		||||
  background-image: linear-gradient(
 | 
			
		||||
    275deg,
 | 
			
		||||
    var(--eve-solar-system-status-friendly),
 | 
			
		||||
    transparent
 | 
			
		||||
  );
 | 
			
		||||
  &.selected {
 | 
			
		||||
    border-color: var(--eve-solar-system-status-color-home);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-friendly {
 | 
			
		||||
  border: 1px solid var(--eve-solar-system-status-color-friendly-dark20);
 | 
			
		||||
  background-image: linear-gradient(
 | 
			
		||||
    275deg,
 | 
			
		||||
    var(--eve-solar-system-status-friendly-dark30),
 | 
			
		||||
    transparent
 | 
			
		||||
  );
 | 
			
		||||
  &.selected {
 | 
			
		||||
    border-color: var(--eve-solar-system-status-color-friendly-dark5);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-lookingFor {
 | 
			
		||||
  border: 1px solid var(--eve-solar-system-status-color-lookingFor-dark15);
 | 
			
		||||
  background-image: linear-gradient(275deg, #45ff8f2f, #457fff2f);
 | 
			
		||||
  &.selected {
 | 
			
		||||
    border-color: var(--pastel-pink, #d291bc);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-warning {
 | 
			
		||||
  background-image: linear-gradient(
 | 
			
		||||
    275deg,
 | 
			
		||||
    var(--eve-solar-system-status-warning),
 | 
			
		||||
    transparent
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-dangerous {
 | 
			
		||||
  background-image: linear-gradient(
 | 
			
		||||
    275deg,
 | 
			
		||||
    var(--eve-solar-system-status-dangerous),
 | 
			
		||||
    transparent
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-target {
 | 
			
		||||
  background-image: linear-gradient(
 | 
			
		||||
    275deg,
 | 
			
		||||
    var(--eve-solar-system-status-target),
 | 
			
		||||
    transparent
 | 
			
		||||
  );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.Bookmarks {
 | 
			
		||||
  position: absolute;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  z-index: 1;
 | 
			
		||||
  display: flex;
 | 
			
		||||
  left: 4px;
 | 
			
		||||
}
 | 
			
		||||
.Bookmark {
 | 
			
		||||
  min-width: 13px;
 | 
			
		||||
  height: 22px;
 | 
			
		||||
  position: relative;
 | 
			
		||||
  top: -13px;
 | 
			
		||||
  border-radius: 5px;
 | 
			
		||||
  color: #ffffff;
 | 
			
		||||
  font-size: 8px;
 | 
			
		||||
  text-align: center;
 | 
			
		||||
  padding-top: 2px;
 | 
			
		||||
  font-weight: bolder;
 | 
			
		||||
  padding-left: 3px;
 | 
			
		||||
  padding-right: 3px;
 | 
			
		||||
  &:not(:first-child) {
 | 
			
		||||
    box-shadow: inset 4px -3px 4px rgba(0, 0, 0, 0.3);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
.BookmarkWithIcon {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: center;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  margin-top: -2px;
 | 
			
		||||
  text-shadow: 0 0 3px rgba(0, 0, 0, 1);
 | 
			
		||||
  padding-right: 2px;
 | 
			
		||||
  .icon {
 | 
			
		||||
    width: 8px;
 | 
			
		||||
    height: 8px;
 | 
			
		||||
    font-size: 8px;
 | 
			
		||||
  }
 | 
			
		||||
  .text {
 | 
			
		||||
    margin-top: 1px;
 | 
			
		||||
    font-size: 9px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
.TagTitle {
 | 
			
		||||
  font-size: 11px;
 | 
			
		||||
  font-weight: bold;
 | 
			
		||||
  text-shadow: 0 0 2px rgba(231, 146, 52, 0.73);
 | 
			
		||||
  color: var(--rf-tag-color, #ffb01d);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* The top row (HeadRow). */
 | 
			
		||||
.HeadRow {
 | 
			
		||||
  position: relative;
 | 
			
		||||
  top: 1px;
 | 
			
		||||
 | 
			
		||||
  .classTitle {
 | 
			
		||||
    font-weight: bold;
 | 
			
		||||
    text-shadow: 0 0 2px rgba(0, 0, 0, 0.73);
 | 
			
		||||
    font-size: 11px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  @-moz-document url-prefix() {
 | 
			
		||||
    .classSystemName {
 | 
			
		||||
      font-family: var(--rf-node-font, inherit);
 | 
			
		||||
      font-weight: bold;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Usually ~19px tall for bottom row. */
 | 
			
		||||
.BottomRow {
 | 
			
		||||
  height: 19px;
 | 
			
		||||
 | 
			
		||||
  .localCounter {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    gap: 2px;
 | 
			
		||||
  }
 | 
			
		||||
  .hasUserCharacters {
 | 
			
		||||
    color: var(--rf-has-user-characters, #fbbf24);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* The systemName overflow + effect icon, etc. */
 | 
			
		||||
.systemNameOverflow {
 | 
			
		||||
  flex-grow: 1;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
  text-overflow: ellipsis;
 | 
			
		||||
  white-space: nowrap;
 | 
			
		||||
  font-family: sans-serif;
 | 
			
		||||
}
 | 
			
		||||
.effect {
 | 
			
		||||
  width: 8px;
 | 
			
		||||
  height: 8px;
 | 
			
		||||
  margin-top: -2px;
 | 
			
		||||
  border-radius: 2px;
 | 
			
		||||
  margin-left: 1px;
 | 
			
		||||
}
 | 
			
		||||
.statics {
 | 
			
		||||
  @-moz-document url-prefix() {
 | 
			
		||||
    position: relative;
 | 
			
		||||
    top: -1px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* The node's invisible handles/double-click overlay. */
 | 
			
		||||
.Handlers {
 | 
			
		||||
  position: absolute;
 | 
			
		||||
  z-index: 2;
 | 
			
		||||
  top: 0;
 | 
			
		||||
  left: 0;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  height: 100%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* The little React Flow handles on each edge. */
 | 
			
		||||
.Handle {
 | 
			
		||||
  border: 1px solid var(--pastel-blue, #5a7d9a);
 | 
			
		||||
  width: 5px;
 | 
			
		||||
  height: 5px;
 | 
			
		||||
 | 
			
		||||
  &.selected {
 | 
			
		||||
    border-color: var(--pastel-pink, #d291bc);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.HandleTop {
 | 
			
		||||
    top: -2px;
 | 
			
		||||
  }
 | 
			
		||||
  &.HandleRight {
 | 
			
		||||
    right: -2px;
 | 
			
		||||
  }
 | 
			
		||||
  &.HandleBottom {
 | 
			
		||||
    bottom: -2px;
 | 
			
		||||
  }
 | 
			
		||||
  &.HandleLeft {
 | 
			
		||||
    left: -2px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.Tick {
 | 
			
		||||
    width: 7px;
 | 
			
		||||
    height: 7px;
 | 
			
		||||
 | 
			
		||||
    &.HandleTop {
 | 
			
		||||
      top: -3px;
 | 
			
		||||
    }
 | 
			
		||||
    &.HandleRight {
 | 
			
		||||
      right: -3px;
 | 
			
		||||
    }
 | 
			
		||||
    &.HandleBottom {
 | 
			
		||||
      bottom: -3px;
 | 
			
		||||
    }
 | 
			
		||||
    &.HandleLeft {
 | 
			
		||||
      left: -3px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Unsplashed signature containers. */
 | 
			
		||||
.Unsplashed {
 | 
			
		||||
  position: absolute;
 | 
			
		||||
  width: calc(50% - 4px);
 | 
			
		||||
  z-index: -1;
 | 
			
		||||
  display: flex;
 | 
			
		||||
  flex-wrap: wrap;
 | 
			
		||||
  gap: 2px;
 | 
			
		||||
  left: 2px;
 | 
			
		||||
  &--right {
 | 
			
		||||
    left: calc(50% + 6px);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,334 +0,0 @@
 | 
			
		||||
import { memo, useMemo } from 'react';
 | 
			
		||||
import { Handle, Position, WrapNodeProps } from 'reactflow';
 | 
			
		||||
import { MapSolarSystemType } from '../../map.types';
 | 
			
		||||
import classes from './SolarSystemNode.module.scss';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
import { useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
import {
 | 
			
		||||
  EFFECT_BACKGROUND_STYLES,
 | 
			
		||||
  LABELS_INFO,
 | 
			
		||||
  LABELS_ORDER,
 | 
			
		||||
  MARKER_BOOKMARK_BG_STYLES,
 | 
			
		||||
  STATUS_CLASSES,
 | 
			
		||||
} from '@/hooks/Mapper/components/map/constants.ts';
 | 
			
		||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace.ts';
 | 
			
		||||
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
 | 
			
		||||
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
 | 
			
		||||
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
 | 
			
		||||
import { getSystemClassStyles, prepareUnsplashedChunks } from '@/hooks/Mapper/components/map/helpers';
 | 
			
		||||
import { sortWHClasses } from '@/hooks/Mapper/helpers';
 | 
			
		||||
import { PrimeIcons } from 'primereact/api';
 | 
			
		||||
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager.ts';
 | 
			
		||||
import { OutCommand } from '@/hooks/Mapper/types';
 | 
			
		||||
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick.ts';
 | 
			
		||||
import { REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
 | 
			
		||||
 | 
			
		||||
const SpaceToClass: Record<string, string> = {
 | 
			
		||||
  [Spaces.Caldari]: classes.Caldaria,
 | 
			
		||||
  [Spaces.Matar]: classes.Mataria,
 | 
			
		||||
  [Spaces.Amarr]: classes.Amarria,
 | 
			
		||||
  [Spaces.Gallente]: classes.Gallente,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const sortedLabels = (labels: string[]) => {
 | 
			
		||||
  if (!labels) {
 | 
			
		||||
    return [];
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return LABELS_ORDER.filter(x => labels.includes(x)).map(x => LABELS_INFO[x]);
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export const getActivityType = (count: number) => {
 | 
			
		||||
  if (count <= 5) {
 | 
			
		||||
    return 'activityNormal';
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (count <= 30) {
 | 
			
		||||
    return 'activityWarn';
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return 'activityDanger';
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// eslint-disable-next-line react/display-name
 | 
			
		||||
export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarSystemType>) => {
 | 
			
		||||
  const { interfaceSettings } = useMapRootState();
 | 
			
		||||
  const { isShowUnsplashedSignatures } = interfaceSettings;
 | 
			
		||||
 | 
			
		||||
  const isTempSystemNameEnabled = useMapGetOption('show_temp_system_name') === 'true';
 | 
			
		||||
 | 
			
		||||
  const {
 | 
			
		||||
    system_class,
 | 
			
		||||
    security,
 | 
			
		||||
    class_title,
 | 
			
		||||
    solar_system_id,
 | 
			
		||||
    statics,
 | 
			
		||||
    effect_name,
 | 
			
		||||
    region_name,
 | 
			
		||||
    region_id,
 | 
			
		||||
    is_shattered,
 | 
			
		||||
    solar_system_name,
 | 
			
		||||
  } = data.system_static_info;
 | 
			
		||||
 | 
			
		||||
  const signatures = data.system_signatures;
 | 
			
		||||
 | 
			
		||||
  const { locked, name, tag, status, labels, id, temporary_name: temporaryName } = data || {};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  const {
 | 
			
		||||
    data: {
 | 
			
		||||
      characters,
 | 
			
		||||
      presentCharacters,
 | 
			
		||||
      wormholesData,
 | 
			
		||||
      hubs,
 | 
			
		||||
      kills,
 | 
			
		||||
      userCharacters,
 | 
			
		||||
      isConnecting,
 | 
			
		||||
      hoverNodeId,
 | 
			
		||||
      visibleNodes,
 | 
			
		||||
      showKSpaceBG,
 | 
			
		||||
      isThickConnections,
 | 
			
		||||
    },
 | 
			
		||||
    outCommand,
 | 
			
		||||
  } = useMapState();
 | 
			
		||||
 | 
			
		||||
  const visible = useMemo(() => visibleNodes.has(id), [id, visibleNodes]);
 | 
			
		||||
 | 
			
		||||
  const charactersInSystem = useMemo(() => {
 | 
			
		||||
    return characters
 | 
			
		||||
      .filter(c => c.location?.solar_system_id === solar_system_id)
 | 
			
		||||
      .filter(c => c.online);
 | 
			
		||||
  }, [characters, presentCharacters, 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 lebM = useMemo(() => new LabelsManager(labels ?? ''), [labels]);
 | 
			
		||||
  const labelsInfo = useMemo(() => sortedLabels(lebM.list), [lebM]);
 | 
			
		||||
  const labelCustom = useMemo(() => lebM.customLabel, [lebM]);
 | 
			
		||||
 | 
			
		||||
  const killsCount = useMemo(() => {
 | 
			
		||||
    const systemKills = kills[solar_system_id];
 | 
			
		||||
    if (!systemKills) {
 | 
			
		||||
      return null;
 | 
			
		||||
    }
 | 
			
		||||
    return systemKills;
 | 
			
		||||
  }, [kills, solar_system_id]);
 | 
			
		||||
 | 
			
		||||
  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 systemName = (isTempSystemNameEnabled && temporaryName) || solar_system_name;
 | 
			
		||||
  const customName = (isTempSystemNameEnabled && temporaryName && name) || (solar_system_name !== name && name);
 | 
			
		||||
 | 
			
		||||
  const [unsplashedLeft, unsplashedRight] = useMemo(() => {
 | 
			
		||||
    if (!isShowUnsplashedSignatures) {
 | 
			
		||||
      return [[], []];
 | 
			
		||||
    }
 | 
			
		||||
    return prepareUnsplashedChunks(
 | 
			
		||||
      signatures
 | 
			
		||||
        .filter(s => s.group === 'Wormhole' && !s.linked_system)
 | 
			
		||||
        .map(s => ({
 | 
			
		||||
          eve_id: s.eve_id,
 | 
			
		||||
          type: s.type,
 | 
			
		||||
          custom_info: s.custom_info,
 | 
			
		||||
        })),
 | 
			
		||||
    );
 | 
			
		||||
  }, [isShowUnsplashedSignatures, signatures]);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <>
 | 
			
		||||
      {visible && (
 | 
			
		||||
        <div className={classes.Bookmarks}>
 | 
			
		||||
          {labelCustom !== '' && (
 | 
			
		||||
            <div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
 | 
			
		||||
              <span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]">{labelCustom}</span>
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
 | 
			
		||||
          {is_shattered && (
 | 
			
		||||
            <div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.shattered)}>
 | 
			
		||||
              <span className={clsx('pi pi-chart-pie text-[0.55rem]')} />
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
 | 
			
		||||
          {killsCount && (
 | 
			
		||||
            <div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[getActivityType(killsCount)])}>
 | 
			
		||||
              <div className={clsx(classes.BookmarkWithIcon)}>
 | 
			
		||||
                <span className={clsx(PrimeIcons.BOLT, 'text-[0.65rem]')} />
 | 
			
		||||
                <span className={clsx(classes.text)}>{killsCount}</span>
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
 | 
			
		||||
          {labelsInfo.map(x => (
 | 
			
		||||
            <div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
 | 
			
		||||
              {x.shortName}
 | 
			
		||||
            </div>
 | 
			
		||||
          ))}
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
      <div
 | 
			
		||||
        onMouseDownCapture={dbClick}
 | 
			
		||||
        className={clsx(
 | 
			
		||||
          classes.RootCustomNode,
 | 
			
		||||
          regionClass,
 | 
			
		||||
          classes[STATUS_CLASSES[status]],
 | 
			
		||||
          { [classes.selected]: selected },
 | 
			
		||||
          'flex flex-col w-[130px] h-[34px]',
 | 
			
		||||
          'px-[6px] pt-[2px] pb-[3px] text-[10px]',
 | 
			
		||||
          'leading-[1] space-y-[1px]',
 | 
			
		||||
          'shadow-[0_0_5px_rgba(45,45,45,0.5)]',
 | 
			
		||||
          'border border-[var(--pastel-blue-darken10)] rounded-[5px]'
 | 
			
		||||
        )}
 | 
			
		||||
      >
 | 
			
		||||
        {visible && (
 | 
			
		||||
          <>
 | 
			
		||||
            <div className={clsx(classes.HeadRow, 'flex items-center gap-[3px]')}>
 | 
			
		||||
              <div className={clsx(classes.classTitle, classTitleColor, '[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]')}>
 | 
			
		||||
                {class_title ?? '-'}
 | 
			
		||||
              </div>
 | 
			
		||||
 | 
			
		||||
              {tag != null && tag !== '' && (
 | 
			
		||||
                <div className={clsx(classes.TagTitle, "color: #38bdf8; font-weight: 500;")}>
 | 
			
		||||
                  {tag}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              <div
 | 
			
		||||
                className={clsx(
 | 
			
		||||
                  classes.classSystemName,
 | 
			
		||||
                  'flex-grow overflow-hidden text-ellipsis whitespace-nowrap font-sans',
 | 
			
		||||
                )}
 | 
			
		||||
              >
 | 
			
		||||
                {systemName}
 | 
			
		||||
              </div>
 | 
			
		||||
 | 
			
		||||
              {isWormhole && (
 | 
			
		||||
                <div className={clsx(classes.statics, 'flex gap-[2px] text-[8px]')}>
 | 
			
		||||
                  {sortedStatics.map(x => (
 | 
			
		||||
                    <WormholeClassComp key={x} id={x} />
 | 
			
		||||
                  ))}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              {effect_name !== null && isWormhole && (
 | 
			
		||||
                <div className={clsx(classes.effect, EFFECT_BACKGROUND_STYLES[effect_name])}></div>
 | 
			
		||||
              )}
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <div className={clsx(classes.BottomRow, 'flex items-center gap-[3px]')}>
 | 
			
		||||
              {customName && (
 | 
			
		||||
                <div className={clsx('font-bold', classes.customName)}>
 | 
			
		||||
                  {customName}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
              {!isWormhole && !customName && <div className={clsx(classes.regionName)}>{region_name}</div>}
 | 
			
		||||
              {isWormhole && !customName && <div />}
 | 
			
		||||
 | 
			
		||||
              <div className="flex items-center ml-auto gap-[2px]">
 | 
			
		||||
                {locked && (
 | 
			
		||||
                  <i className={clsx(PrimeIcons.LOCK, 'text-[0.45rem] font-bold')} />
 | 
			
		||||
                )}
 | 
			
		||||
                {hubs.includes(solar_system_id.toString()) && (
 | 
			
		||||
                  <i className={clsx(PrimeIcons.MAP_MARKER, 'text-[0.45rem] font-bold')} />
 | 
			
		||||
                )}
 | 
			
		||||
                {charactersInSystem.length > 0 && (
 | 
			
		||||
                  <div
 | 
			
		||||
                    className={clsx(
 | 
			
		||||
                      classes.localCounter,
 | 
			
		||||
                      { [classes.hasUserCharacters]: hasUserCharacters },
 | 
			
		||||
                      'flex gap-[2px]'
 | 
			
		||||
                    )}
 | 
			
		||||
                  >
 | 
			
		||||
                    <i className="pi pi-users text-[0.50rem]" />
 | 
			
		||||
                    <span className="text-[0.65rem]">{charactersInSystem.length}</span>
 | 
			
		||||
                  </div>
 | 
			
		||||
                )}
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </>
 | 
			
		||||
        )}
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      {visible && isShowUnsplashedSignatures && (
 | 
			
		||||
        <div className={classes.Unsplashed}>
 | 
			
		||||
          {unsplashedLeft.map(x => (
 | 
			
		||||
            <UnsplashedSignature key={x.sig_id} signature={x} />
 | 
			
		||||
          ))}
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
      {visible && isShowUnsplashedSignatures && (
 | 
			
		||||
        <div className={clsx(classes.Unsplashed, classes['Unsplashed--right'])}>
 | 
			
		||||
          {unsplashedRight.map(x => (
 | 
			
		||||
            <UnsplashedSignature key={x.sig_id} signature={x} />
 | 
			
		||||
          ))}
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
 | 
			
		||||
      <div className={classes.Handlers}>
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleTop, {
 | 
			
		||||
            [classes.selected]: selected,
 | 
			
		||||
            [classes.Tick]: isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Top}
 | 
			
		||||
          id="a"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleRight, {
 | 
			
		||||
            [classes.selected]: selected,
 | 
			
		||||
            [classes.Tick]: isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Right}
 | 
			
		||||
          id="b"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleBottom, {
 | 
			
		||||
            [classes.selected]: selected,
 | 
			
		||||
            [classes.Tick]: isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Bottom}
 | 
			
		||||
          id="c"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleLeft, {
 | 
			
		||||
            [classes.selected]: selected,
 | 
			
		||||
            [classes.Tick]: isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Left}
 | 
			
		||||
          id="d"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
    </>
 | 
			
		||||
  );
 | 
			
		||||
});
 | 
			
		||||
@@ -0,0 +1,369 @@
 | 
			
		||||
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
 | 
			
		||||
 | 
			
		||||
$pastel-blue: #5a7d9a;
 | 
			
		||||
$pastel-pink: #d291bc;
 | 
			
		||||
$dark-bg: #2d2d2d;
 | 
			
		||||
$text-color: #ffffff;
 | 
			
		||||
$tooltip-bg: #202020;
 | 
			
		||||
 | 
			
		||||
.RootCustomNode {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  width: 130px;
 | 
			
		||||
  height: 34px;
 | 
			
		||||
 | 
			
		||||
  font-family: var(--rf-node-font-family, inherit) !important;
 | 
			
		||||
  font-weight: var(--rf-node-font-weight, inherit) !important;
 | 
			
		||||
 | 
			
		||||
  flex-direction: column;
 | 
			
		||||
  padding: 2px 6px;
 | 
			
		||||
  font-size: 10px;
 | 
			
		||||
 | 
			
		||||
  background-color: var(--rf-node-bg-color, #202020) !important;
 | 
			
		||||
  color: var(--rf-text-color, #ffffff);
 | 
			
		||||
 | 
			
		||||
  box-shadow: 0 0 5px rgba($dark-bg, 0.5);
 | 
			
		||||
  border: 1px solid darken($pastel-blue, 10%);
 | 
			
		||||
  border-radius: 5px;
 | 
			
		||||
  position: relative;
 | 
			
		||||
  z-index: 3;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
 | 
			
		||||
  &.Mataria,
 | 
			
		||||
  &.Amarria,
 | 
			
		||||
  &.Gallente,
 | 
			
		||||
  &.Caldaria {
 | 
			
		||||
    &::before {
 | 
			
		||||
      content: '';
 | 
			
		||||
      position: absolute;
 | 
			
		||||
      top: 0;
 | 
			
		||||
      left: 0;
 | 
			
		||||
      right: 0;
 | 
			
		||||
      bottom: 0;
 | 
			
		||||
      background-size: cover;
 | 
			
		||||
      background-position: 50% 50%;
 | 
			
		||||
      z-index: -1;
 | 
			
		||||
      background-repeat: no-repeat;
 | 
			
		||||
      border-radius: 3px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.Mataria {
 | 
			
		||||
    &::before {
 | 
			
		||||
      background-image: url('/images/mataria-180.png');
 | 
			
		||||
      opacity: 0.6;
 | 
			
		||||
      background-position-x: 1px;
 | 
			
		||||
      background-position-y: -14px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.Caldaria {
 | 
			
		||||
    &::before {
 | 
			
		||||
      background-image: url('/images/caldaria-180.png');
 | 
			
		||||
      opacity: 0.6;
 | 
			
		||||
      background-position-x: 1px;
 | 
			
		||||
      background-position-y: -10px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.Amarria {
 | 
			
		||||
    &::before {
 | 
			
		||||
      opacity: 0.45;
 | 
			
		||||
      background-image: url('/images/amarr-180.png');
 | 
			
		||||
      background-position-x: 0;
 | 
			
		||||
      background-position-y: -13px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.Gallente {
 | 
			
		||||
    &::before {
 | 
			
		||||
      opacity: 0.5;
 | 
			
		||||
      background-image: url('/images/gallente-180.png');
 | 
			
		||||
      background-position-x: 1px;
 | 
			
		||||
      background-position-y: 0;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.selected {
 | 
			
		||||
    border-color: $pastel-pink;
 | 
			
		||||
    box-shadow: 0 0 10px #9a1af1c2;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.eve-system-status-home {
 | 
			
		||||
    border: 1px solid var(--eve-solar-system-status-color-home-dark30);
 | 
			
		||||
    background-image: linear-gradient(45deg, var(--eve-solar-system-status-color-background), transparent);
 | 
			
		||||
    &.selected {
 | 
			
		||||
      border-color: var(--eve-solar-system-status-color-home);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.eve-system-status-friendly {
 | 
			
		||||
    border: 1px solid var(--eve-solar-system-status-color-friendly-dark20);
 | 
			
		||||
    background-image: linear-gradient(275deg, var(--eve-solar-system-status-friendly-dark30), transparent);
 | 
			
		||||
    &.selected {
 | 
			
		||||
      border-color: var(--eve-solar-system-status-color-friendly-dark5);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.eve-system-status-lookingFor {
 | 
			
		||||
    border: 1px solid var(--eve-solar-system-status-color-lookingFor-dark15);
 | 
			
		||||
    background-image: linear-gradient(275deg, #45ff8f2f, #457fff2f);
 | 
			
		||||
    &.selected {
 | 
			
		||||
      border-color: $pastel-pink;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.eve-system-status-warning {
 | 
			
		||||
    background-image: linear-gradient(275deg, var(--eve-solar-system-status-warning), transparent);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.eve-system-status-dangerous {
 | 
			
		||||
    background-image: linear-gradient(275deg, var(--eve-solar-system-status-dangerous), transparent);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.eve-system-status-target {
 | 
			
		||||
    background-image: linear-gradient(275deg, var(--eve-solar-system-status-target), transparent);
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.Bookmarks {
 | 
			
		||||
  position: absolute;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  z-index: 1;
 | 
			
		||||
  display: flex;
 | 
			
		||||
  left: 4px;
 | 
			
		||||
 | 
			
		||||
  & > .Bookmark {
 | 
			
		||||
    min-width: 13px;
 | 
			
		||||
    height: 22px;
 | 
			
		||||
    position: relative;
 | 
			
		||||
    top: -13px;
 | 
			
		||||
    border-radius: 5px;
 | 
			
		||||
    color: #ffffff;
 | 
			
		||||
    font-size: 8px;
 | 
			
		||||
    text-align: center;
 | 
			
		||||
    padding-top: 2px;
 | 
			
		||||
    font-weight: bolder;
 | 
			
		||||
    padding-left: 3px;
 | 
			
		||||
    padding-right: 3px;
 | 
			
		||||
 | 
			
		||||
    &:not(:first-child) {
 | 
			
		||||
      box-shadow: inset 4px -3px 4px rgba(0, 0, 0, 0.3);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .BookmarkWithIcon {
 | 
			
		||||
    display: flex;
 | 
			
		||||
    justify-content: center;
 | 
			
		||||
    align-items: center;
 | 
			
		||||
    margin-top: -2px;
 | 
			
		||||
    text-shadow: 0 0 3px rgba(0, 0, 0, 1);
 | 
			
		||||
    padding-right: 2px;
 | 
			
		||||
 | 
			
		||||
    & > .icon {
 | 
			
		||||
      width: 8px;
 | 
			
		||||
      height: 8px;
 | 
			
		||||
      font-size: 8px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    & > .text {
 | 
			
		||||
      margin-top: 1px;
 | 
			
		||||
      font-size: 9px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.Unsplashed {
 | 
			
		||||
  position: absolute;
 | 
			
		||||
  width: calc(50% - 4px);
 | 
			
		||||
  z-index: -1;
 | 
			
		||||
  display: flex;
 | 
			
		||||
  flex-wrap: wrap;
 | 
			
		||||
  gap: 2px;
 | 
			
		||||
  left: 2px;
 | 
			
		||||
 | 
			
		||||
  &--right {
 | 
			
		||||
    left: calc(50% + 6px);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  & > .Signature {
 | 
			
		||||
    width: 13px;
 | 
			
		||||
    height: 4px;
 | 
			
		||||
    position: relative;
 | 
			
		||||
    top: 3px;
 | 
			
		||||
    border-radius: 5px;
 | 
			
		||||
    color: #ffffff;
 | 
			
		||||
    font-size: 8px;
 | 
			
		||||
    text-align: center;
 | 
			
		||||
    padding-top: 2px;
 | 
			
		||||
    font-weight: bolder;
 | 
			
		||||
    padding-left: 3px;
 | 
			
		||||
    padding-right: 3px;
 | 
			
		||||
    display: block;
 | 
			
		||||
 | 
			
		||||
    background-color: #833ca4;
 | 
			
		||||
 | 
			
		||||
    &:not(:first-child) {
 | 
			
		||||
      box-shadow: inset 4px -3px 4px rgba(0, 0, 0, 0.3);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.icon {
 | 
			
		||||
  width: 8px;
 | 
			
		||||
  height: 8px;
 | 
			
		||||
  font-size: 8px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.HeadRow {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  gap: 3px;
 | 
			
		||||
  font-size: 11px;
 | 
			
		||||
  line-height: 14px;
 | 
			
		||||
  font-weight: 500;
 | 
			
		||||
  position: relative;
 | 
			
		||||
  top: 1px;
 | 
			
		||||
 | 
			
		||||
  .classTitle {
 | 
			
		||||
    font-size: 11px;
 | 
			
		||||
    font-weight: bold;
 | 
			
		||||
    text-shadow: 0 0 2px rgb(0 0 0 / 73%);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .TagTitle {
 | 
			
		||||
    font-size: 11px;
 | 
			
		||||
    font-weight: 500;
 | 
			
		||||
    text-shadow: 0 0 2px rgba(231, 146, 52, 0.73);
 | 
			
		||||
    color: var(--rf-tag-color, #38bdf8);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /* Firefox kostyl */
 | 
			
		||||
  @-moz-document url-prefix() {
 | 
			
		||||
    .classSystemName {
 | 
			
		||||
      font-weight: bold;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.BottomRow {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  justify-content: space-between;
 | 
			
		||||
  align-items: center;
 | 
			
		||||
  height: 19px;
 | 
			
		||||
 | 
			
		||||
  .hasLocalCounter {
 | 
			
		||||
    margin-right: 2px;
 | 
			
		||||
    &.countAbove9 {
 | 
			
		||||
      margin-right: 1.5rem;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .lockIcon {
 | 
			
		||||
    font-size: 0.45rem;
 | 
			
		||||
    font-weight: bold;
 | 
			
		||||
    position: relative;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .mapMarker {
 | 
			
		||||
    font-size: 0.45rem;
 | 
			
		||||
    font-weight: bold;
 | 
			
		||||
    position: relative;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.effect {
 | 
			
		||||
  width: 8px;
 | 
			
		||||
  height: 8px;
 | 
			
		||||
  margin-top: -2px;
 | 
			
		||||
  box-sizing: border-box;
 | 
			
		||||
  border-radius: 2px;
 | 
			
		||||
  margin-left: 1px;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.statics {
 | 
			
		||||
  display: flex;
 | 
			
		||||
  gap: 2px;
 | 
			
		||||
  font-size: 8px;
 | 
			
		||||
 | 
			
		||||
  & > * {
 | 
			
		||||
    line-height: 10px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /* Firefox kostyl */
 | 
			
		||||
  @-moz-document url-prefix() {
 | 
			
		||||
    position: relative;
 | 
			
		||||
    top: -1px;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.Handlers {
 | 
			
		||||
  position: absolute;
 | 
			
		||||
  z-index: 4;
 | 
			
		||||
  pointer-events: none;
 | 
			
		||||
  top: 0;
 | 
			
		||||
  left: 0;
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  height: 100%;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.Handle {
 | 
			
		||||
  min-width: initial;
 | 
			
		||||
  min-height: initial;
 | 
			
		||||
  border: 1px solid $pastel-blue;
 | 
			
		||||
  width: 5px;
 | 
			
		||||
  height: 5px;
 | 
			
		||||
  pointer-events: auto;
 | 
			
		||||
 | 
			
		||||
  &.selected {
 | 
			
		||||
    border-color: $pastel-pink;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.HandleTop {
 | 
			
		||||
    top: -2px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.HandleRight {
 | 
			
		||||
    right: -2px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.HandleBottom {
 | 
			
		||||
    bottom: -2px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.HandleLeft {
 | 
			
		||||
    left: -2px;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  &.Tick {
 | 
			
		||||
    width: 7px;
 | 
			
		||||
    height: 7px;
 | 
			
		||||
 | 
			
		||||
    &.HandleTop {
 | 
			
		||||
      top: -3px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    &.HandleRight {
 | 
			
		||||
      right: -3px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    &.HandleBottom {
 | 
			
		||||
      bottom: -3px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    &.HandleLeft {
 | 
			
		||||
      left: -3px;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.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)
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,221 @@
 | 
			
		||||
import { memo } from 'react';
 | 
			
		||||
import { MapSolarSystemType } from '../../map.types';
 | 
			
		||||
import { Handle, NodeProps, Position } from 'reactflow';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
import classes from './SolarSystemNodeDefault.module.scss';
 | 
			
		||||
import { PrimeIcons } from 'primereact/api';
 | 
			
		||||
import { useLocalCounter, useNodeKillsCount, useSolarSystemNode } from '../../hooks';
 | 
			
		||||
import {
 | 
			
		||||
  EFFECT_BACKGROUND_STYLES,
 | 
			
		||||
  MARKER_BOOKMARK_BG_STYLES,
 | 
			
		||||
  STATUS_CLASSES,
 | 
			
		||||
} from '@/hooks/Mapper/components/map/constants';
 | 
			
		||||
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
 | 
			
		||||
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
 | 
			
		||||
import { LocalCounter } from './SolarSystemLocalCounter';
 | 
			
		||||
import { KillsCounter } from './SolarSystemKillsCounter';
 | 
			
		||||
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
 | 
			
		||||
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
 | 
			
		||||
import { Tag } from 'primereact/tag';
 | 
			
		||||
 | 
			
		||||
// let render = 0;
 | 
			
		||||
export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>) => {
 | 
			
		||||
  const nodeVars = useSolarSystemNode(props);
 | 
			
		||||
  const { localCounterCharacters } = useLocalCounter(nodeVars);
 | 
			
		||||
  const localKillsCount = useNodeKillsCount(nodeVars.solarSystemId, nodeVars.killsCount);
 | 
			
		||||
 | 
			
		||||
  // console.log('JOipP', `render ${nodeVars.id}`, render++);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <>
 | 
			
		||||
      {nodeVars.visible && (
 | 
			
		||||
        <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 && (
 | 
			
		||||
            <KillsCounter
 | 
			
		||||
              killsCount={localKillsCount}
 | 
			
		||||
              systemId={nodeVars.solarSystemId}
 | 
			
		||||
              size={TooltipSize.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.labelCustom !== '' && (
 | 
			
		||||
            <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>
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
 | 
			
		||||
          {nodeVars.labelsInfo.map(x => (
 | 
			
		||||
            <div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
 | 
			
		||||
              {x.shortName}
 | 
			
		||||
            </div>
 | 
			
		||||
          ))}
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        className={clsx(
 | 
			
		||||
          classes.RootCustomNode,
 | 
			
		||||
          nodeVars.regionClass && classes[nodeVars.regionClass],
 | 
			
		||||
          nodeVars.status !== undefined ? classes[STATUS_CLASSES[nodeVars.status]] : '',
 | 
			
		||||
          { [classes.selected]: nodeVars.selected },
 | 
			
		||||
        )}
 | 
			
		||||
        onMouseDownCapture={e => nodeVars.dbClick(e)}
 | 
			
		||||
      >
 | 
			
		||||
        {nodeVars.visible && (
 | 
			
		||||
          <>
 | 
			
		||||
            <div className={classes.HeadRow}>
 | 
			
		||||
              <div
 | 
			
		||||
                className={clsx(
 | 
			
		||||
                  classes.classTitle,
 | 
			
		||||
                  nodeVars.classTitleColor,
 | 
			
		||||
                  '[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]',
 | 
			
		||||
                )}
 | 
			
		||||
              >
 | 
			
		||||
                {nodeVars.classTitle ?? '-'}
 | 
			
		||||
              </div>
 | 
			
		||||
 | 
			
		||||
              {nodeVars.tag != null && nodeVars.tag !== '' && (
 | 
			
		||||
                <Tag
 | 
			
		||||
                  value={nodeVars.tag}
 | 
			
		||||
                  severity="warning"
 | 
			
		||||
                  className="py-0 px-[2px] text-[9px] [&_.p-tag-value]:leading-[1.3]"
 | 
			
		||||
                ></Tag>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              <div
 | 
			
		||||
                className={clsx(
 | 
			
		||||
                  classes.classSystemName,
 | 
			
		||||
                  '[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] flex-grow overflow-hidden text-ellipsis whitespace-nowrap font-sans',
 | 
			
		||||
                )}
 | 
			
		||||
              >
 | 
			
		||||
                {nodeVars.systemName}
 | 
			
		||||
              </div>
 | 
			
		||||
 | 
			
		||||
              {nodeVars.isWormhole && (
 | 
			
		||||
                <div className={classes.statics}>
 | 
			
		||||
                  {nodeVars.sortedStatics.map(whClass => (
 | 
			
		||||
                    <WormholeClassComp key={String(whClass)} id={String(whClass)} />
 | 
			
		||||
                  ))}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              {nodeVars.effectName !== null && nodeVars.isWormhole && (
 | 
			
		||||
                <div className={clsx(classes.effect, EFFECT_BACKGROUND_STYLES[nodeVars.effectName])} />
 | 
			
		||||
              )}
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
 | 
			
		||||
              {nodeVars.customName && (
 | 
			
		||||
                <div className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] text-blue-300 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
 | 
			
		||||
                  {nodeVars.customName}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              {!nodeVars.isWormhole && !nodeVars.customName && (
 | 
			
		||||
                <div className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] text-stone-300 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
 | 
			
		||||
                  {nodeVars.regionName}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              {nodeVars.isWormhole && !nodeVars.customName && <div />}
 | 
			
		||||
 | 
			
		||||
              <div className="flex items-center gap-1 justify-end">
 | 
			
		||||
                <div className={clsx('flex items-center gap-1')}>
 | 
			
		||||
                  {nodeVars.locked && <i className={clsx(PrimeIcons.LOCK, classes.lockIcon)} />}
 | 
			
		||||
                  {nodeVars.hubs.includes(nodeVars.solarSystemId) && (
 | 
			
		||||
                    <i className={clsx(PrimeIcons.MAP_MARKER, classes.mapMarker)} />
 | 
			
		||||
                  )}
 | 
			
		||||
                </div>
 | 
			
		||||
 | 
			
		||||
                <LocalCounter
 | 
			
		||||
                  hasUserCharacters={nodeVars.hasUserCharacters}
 | 
			
		||||
                  localCounterCharacters={localCounterCharacters}
 | 
			
		||||
                />
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </>
 | 
			
		||||
        )}
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      {nodeVars.visible && (
 | 
			
		||||
        <>
 | 
			
		||||
          {nodeVars.unsplashedLeft.length > 0 && (
 | 
			
		||||
            <div className={classes.Unsplashed}>
 | 
			
		||||
              {nodeVars.unsplashedLeft.map(sig => (
 | 
			
		||||
                <UnsplashedSignature key={sig.eve_id} signature={sig} />
 | 
			
		||||
              ))}
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
 | 
			
		||||
          {nodeVars.unsplashedRight.length > 0 && (
 | 
			
		||||
            <div className={clsx(classes.Unsplashed, classes['Unsplashed--right'])}>
 | 
			
		||||
              {nodeVars.unsplashedRight.map(sig => (
 | 
			
		||||
                <UnsplashedSignature key={sig.eve_id} signature={sig} />
 | 
			
		||||
              ))}
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
        </>
 | 
			
		||||
      )}
 | 
			
		||||
 | 
			
		||||
      <div className={classes.Handlers}>
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleTop, {
 | 
			
		||||
            [classes.selected]: nodeVars.selected,
 | 
			
		||||
            [classes.Tick]: nodeVars.isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Top}
 | 
			
		||||
          id="a"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleRight, {
 | 
			
		||||
            [classes.selected]: nodeVars.selected,
 | 
			
		||||
            [classes.Tick]: nodeVars.isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Right}
 | 
			
		||||
          id="b"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleBottom, {
 | 
			
		||||
            [classes.selected]: nodeVars.selected,
 | 
			
		||||
            [classes.Tick]: nodeVars.isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Bottom}
 | 
			
		||||
          id="c"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleLeft, {
 | 
			
		||||
            [classes.selected]: nodeVars.selected,
 | 
			
		||||
            [classes.Tick]: nodeVars.isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Left}
 | 
			
		||||
          id="d"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
    </>
 | 
			
		||||
  );
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
SolarSystemNodeDefault.displayName = 'SolarSystemNodeDefault';
 | 
			
		||||
@@ -0,0 +1,20 @@
 | 
			
		||||
@import './SolarSystemNodeDefault.module.scss';
 | 
			
		||||
 | 
			
		||||
/* ---------------------------------------------
 | 
			
		||||
   Only override what's different from the base
 | 
			
		||||
   Currently none required
 | 
			
		||||
---------------------------------------------- */
 | 
			
		||||
 | 
			
		||||
.RootCustomNode {
 | 
			
		||||
  &.eve-system-status-home {
 | 
			
		||||
    border: 1px solid var(--eve-solar-system-status-color-home-dark30);
 | 
			
		||||
    background-image: linear-gradient(
 | 
			
		||||
        275deg,
 | 
			
		||||
        var(--eve-solar-system-status-home),
 | 
			
		||||
        transparent
 | 
			
		||||
    );
 | 
			
		||||
    &.selected {
 | 
			
		||||
      border-color: var(--eve-solar-system-status-color-home);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,226 @@
 | 
			
		||||
import { memo } from 'react';
 | 
			
		||||
import { MapSolarSystemType } from '../../map.types';
 | 
			
		||||
import { Handle, NodeProps, Position } from 'reactflow';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
import classes from './SolarSystemNodeTheme.module.scss';
 | 
			
		||||
import { PrimeIcons } from 'primereact/api';
 | 
			
		||||
import { useLocalCounter, useNodeKillsCount, useSolarSystemNode } from '../../hooks';
 | 
			
		||||
import {
 | 
			
		||||
  EFFECT_BACKGROUND_STYLES,
 | 
			
		||||
  MARKER_BOOKMARK_BG_STYLES,
 | 
			
		||||
  STATUS_CLASSES,
 | 
			
		||||
} from '@/hooks/Mapper/components/map/constants';
 | 
			
		||||
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
 | 
			
		||||
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
 | 
			
		||||
import { LocalCounter } from './SolarSystemLocalCounter';
 | 
			
		||||
import { KillsCounter } from './SolarSystemKillsCounter';
 | 
			
		||||
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
 | 
			
		||||
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
 | 
			
		||||
 | 
			
		||||
// let render = 0;
 | 
			
		||||
export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>) => {
 | 
			
		||||
  const nodeVars = useSolarSystemNode(props);
 | 
			
		||||
  const { localCounterCharacters } = useLocalCounter(nodeVars);
 | 
			
		||||
  const localKillsCount = useNodeKillsCount(nodeVars.solarSystemId, nodeVars.killsCount);
 | 
			
		||||
 | 
			
		||||
  // console.log('JOipP', `render ${nodeVars.id}`, render++);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <>
 | 
			
		||||
      {nodeVars.visible && (
 | 
			
		||||
        <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 && (
 | 
			
		||||
            <KillsCounter
 | 
			
		||||
              killsCount={localKillsCount}
 | 
			
		||||
              systemId={nodeVars.solarSystemId}
 | 
			
		||||
              size={TooltipSize.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.labelCustom !== '' && (
 | 
			
		||||
            <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>
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
 | 
			
		||||
          {nodeVars.labelsInfo.map(x => (
 | 
			
		||||
            <div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
 | 
			
		||||
              {x.shortName}
 | 
			
		||||
            </div>
 | 
			
		||||
          ))}
 | 
			
		||||
        </div>
 | 
			
		||||
      )}
 | 
			
		||||
 | 
			
		||||
      <div
 | 
			
		||||
        className={clsx(
 | 
			
		||||
          classes.RootCustomNode,
 | 
			
		||||
          nodeVars.regionClass && classes[nodeVars.regionClass],
 | 
			
		||||
          nodeVars.status !== undefined ? classes[STATUS_CLASSES[nodeVars.status]] : '',
 | 
			
		||||
          { [classes.selected]: nodeVars.selected },
 | 
			
		||||
        )}
 | 
			
		||||
        onMouseDownCapture={e => nodeVars.dbClick(e)}
 | 
			
		||||
      >
 | 
			
		||||
        {nodeVars.visible && (
 | 
			
		||||
          <>
 | 
			
		||||
            <div className={classes.HeadRow}>
 | 
			
		||||
              <div
 | 
			
		||||
                className={clsx(
 | 
			
		||||
                  classes.classTitle,
 | 
			
		||||
                  nodeVars.classTitleColor,
 | 
			
		||||
                  '[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]',
 | 
			
		||||
                )}
 | 
			
		||||
              >
 | 
			
		||||
                {nodeVars.classTitle ?? '-'}
 | 
			
		||||
              </div>
 | 
			
		||||
 | 
			
		||||
              {nodeVars.tag != null && nodeVars.tag !== '' && (
 | 
			
		||||
                <div className={clsx(classes.TagTitle)}>{nodeVars.tag}</div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              <div
 | 
			
		||||
                className={clsx(
 | 
			
		||||
                  classes.classSystemName,
 | 
			
		||||
                  '[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] flex-grow overflow-hidden text-ellipsis whitespace-nowrap',
 | 
			
		||||
                )}
 | 
			
		||||
              >
 | 
			
		||||
                {nodeVars.systemName}
 | 
			
		||||
              </div>
 | 
			
		||||
 | 
			
		||||
              {nodeVars.isWormhole && (
 | 
			
		||||
                <div className={classes.statics}>
 | 
			
		||||
                  {nodeVars.sortedStatics.map(whClass => (
 | 
			
		||||
                    <WormholeClassComp key={String(whClass)} id={String(whClass)} />
 | 
			
		||||
                  ))}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              {nodeVars.effectName !== null && nodeVars.isWormhole && (
 | 
			
		||||
                <div className={clsx(classes.effect, EFFECT_BACKGROUND_STYLES[nodeVars.effectName])} />
 | 
			
		||||
              )}
 | 
			
		||||
            </div>
 | 
			
		||||
 | 
			
		||||
            <div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
 | 
			
		||||
              {nodeVars.customName && (
 | 
			
		||||
                <div
 | 
			
		||||
                  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}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              {!nodeVars.isWormhole && !nodeVars.customName && (
 | 
			
		||||
                <div
 | 
			
		||||
                  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}
 | 
			
		||||
                </div>
 | 
			
		||||
              )}
 | 
			
		||||
 | 
			
		||||
              {nodeVars.isWormhole && !nodeVars.customName && <div />}
 | 
			
		||||
 | 
			
		||||
              <div className="flex items-center gap-1 justify-end">
 | 
			
		||||
                <div className={clsx('flex items-center gap-1')}>
 | 
			
		||||
                  {nodeVars.locked && <i className={clsx(PrimeIcons.LOCK, classes.lockIcon)} />}
 | 
			
		||||
                  {nodeVars.hubs.includes(nodeVars.solarSystemId) && (
 | 
			
		||||
                    <i className={clsx(PrimeIcons.MAP_MARKER, classes.mapMarker)} />
 | 
			
		||||
                  )}
 | 
			
		||||
                </div>
 | 
			
		||||
 | 
			
		||||
                <LocalCounter
 | 
			
		||||
                  hasUserCharacters={nodeVars.hasUserCharacters}
 | 
			
		||||
                  localCounterCharacters={localCounterCharacters}
 | 
			
		||||
                />
 | 
			
		||||
              </div>
 | 
			
		||||
            </div>
 | 
			
		||||
          </>
 | 
			
		||||
        )}
 | 
			
		||||
      </div>
 | 
			
		||||
 | 
			
		||||
      {nodeVars.visible && (
 | 
			
		||||
        <>
 | 
			
		||||
          {nodeVars.unsplashedLeft.length > 0 && (
 | 
			
		||||
            <div className={classes.Unsplashed}>
 | 
			
		||||
              {nodeVars.unsplashedLeft.map(sig => (
 | 
			
		||||
                <UnsplashedSignature key={sig.eve_id} signature={sig} />
 | 
			
		||||
              ))}
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
 | 
			
		||||
          {nodeVars.unsplashedRight.length > 0 && (
 | 
			
		||||
            <div className={clsx(classes.Unsplashed, classes['Unsplashed--right'])}>
 | 
			
		||||
              {nodeVars.unsplashedRight.map(sig => (
 | 
			
		||||
                <UnsplashedSignature key={sig.eve_id} signature={sig} />
 | 
			
		||||
              ))}
 | 
			
		||||
            </div>
 | 
			
		||||
          )}
 | 
			
		||||
        </>
 | 
			
		||||
      )}
 | 
			
		||||
 | 
			
		||||
      <div className={classes.Handlers}>
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleTop, {
 | 
			
		||||
            [classes.selected]: nodeVars.selected,
 | 
			
		||||
            [classes.Tick]: nodeVars.isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Top}
 | 
			
		||||
          id="a"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleRight, {
 | 
			
		||||
            [classes.selected]: nodeVars.selected,
 | 
			
		||||
            [classes.Tick]: nodeVars.isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Right}
 | 
			
		||||
          id="b"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleBottom, {
 | 
			
		||||
            [classes.selected]: nodeVars.selected,
 | 
			
		||||
            [classes.Tick]: nodeVars.isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Bottom}
 | 
			
		||||
          id="c"
 | 
			
		||||
        />
 | 
			
		||||
        <Handle
 | 
			
		||||
          type="source"
 | 
			
		||||
          className={clsx(classes.Handle, classes.HandleLeft, {
 | 
			
		||||
            [classes.selected]: nodeVars.selected,
 | 
			
		||||
            [classes.Tick]: nodeVars.isThickConnections,
 | 
			
		||||
          })}
 | 
			
		||||
          style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
 | 
			
		||||
          position={Position.Left}
 | 
			
		||||
          id="d"
 | 
			
		||||
        />
 | 
			
		||||
      </div>
 | 
			
		||||
    </>
 | 
			
		||||
  );
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
SolarSystemNodeTheme.displayName = 'SolarSystemNodeTheme';
 | 
			
		||||
@@ -1 +1,2 @@
 | 
			
		||||
export * from './SolarSystemNode';
 | 
			
		||||
export * from './SolarSystemNodeDefault';
 | 
			
		||||
export * from './SolarSystemNodeTheme';
 | 
			
		||||
 
 | 
			
		||||
@@ -15,4 +15,8 @@
 | 
			
		||||
    font-weight: bolder;
 | 
			
		||||
    display: block;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  & > .Eol {
 | 
			
		||||
    display: block;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -8,8 +8,8 @@ import { WORMHOLE_CLASS_STYLES, WORMHOLES_ADDITIONAL_INFO } from '@/hooks/Mapper
 | 
			
		||||
import { useMemo } from 'react';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
import { renderInfoColumn } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/renders';
 | 
			
		||||
 | 
			
		||||
import { k162Types } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureK162TypeSelect';
 | 
			
		||||
import { K162_TYPES_MAP } from '@/hooks/Mapper/constants.ts';
 | 
			
		||||
import { parseSignatureCustomInfo } from '@/hooks/Mapper/helpers/parseSignatureCustomInfo.ts';
 | 
			
		||||
 | 
			
		||||
interface UnsplashedSignatureProps {
 | 
			
		||||
  signature: SystemSignature;
 | 
			
		||||
@@ -22,17 +22,22 @@ export const UnsplashedSignature = ({ signature }: UnsplashedSignatureProps) =>
 | 
			
		||||
  const whData = useMemo(() => wormholesData[signature.type], [signature.type, wormholesData]);
 | 
			
		||||
  const whClass = useMemo(() => (whData ? WORMHOLES_ADDITIONAL_INFO[whData.dest] : null), [whData]);
 | 
			
		||||
 | 
			
		||||
  const k162TypeOption = useMemo(() => {
 | 
			
		||||
    if (!signature.custom_info) {
 | 
			
		||||
      return null;
 | 
			
		||||
    }
 | 
			
		||||
    const customInfo = JSON.parse(signature.custom_info);
 | 
			
		||||
    if (!customInfo.k162Type) {
 | 
			
		||||
      return null;
 | 
			
		||||
    }
 | 
			
		||||
    return k162Types.find(x => x.value === customInfo.k162Type);
 | 
			
		||||
  const customInfo = useMemo(() => {
 | 
			
		||||
    return parseSignatureCustomInfo(signature.custom_info);
 | 
			
		||||
  }, [signature]);
 | 
			
		||||
 | 
			
		||||
  const k162TypeOption = useMemo(() => {
 | 
			
		||||
    if (!customInfo?.k162Type) {
 | 
			
		||||
      return null;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    return K162_TYPES_MAP[customInfo.k162Type];
 | 
			
		||||
  }, [customInfo]);
 | 
			
		||||
 | 
			
		||||
  const isEOL = useMemo(() => {
 | 
			
		||||
    return customInfo?.isEOL;
 | 
			
		||||
  }, [customInfo]);
 | 
			
		||||
 | 
			
		||||
  const whClassStyle = useMemo(() => {
 | 
			
		||||
    if (signature.type === 'K162' && k162TypeOption) {
 | 
			
		||||
      const k162Data = wormholesData[k162TypeOption.whClassName];
 | 
			
		||||
@@ -45,19 +50,19 @@ export const UnsplashedSignature = ({ signature }: UnsplashedSignatureProps) =>
 | 
			
		||||
  return (
 | 
			
		||||
    <WdTooltipWrapper
 | 
			
		||||
      className={clsx(classes.Signature)}
 | 
			
		||||
      // @ts-ignore
 | 
			
		||||
      content={
 | 
			
		||||
        (
 | 
			
		||||
          <div className="flex flex-col gap-1">
 | 
			
		||||
            <InfoDrawer title={<b className="text-slate-50">{signature.eve_id}</b>}>
 | 
			
		||||
              {renderInfoColumn(signature)}
 | 
			
		||||
            </InfoDrawer>
 | 
			
		||||
          </div>
 | 
			
		||||
        ) as React.ReactNode
 | 
			
		||||
        <div className="flex flex-col gap-1">
 | 
			
		||||
          <InfoDrawer title={<b className="text-slate-50">{signature.eve_id}</b>}>
 | 
			
		||||
            {renderInfoColumn(signature)}
 | 
			
		||||
          </InfoDrawer>
 | 
			
		||||
        </div>
 | 
			
		||||
      }
 | 
			
		||||
    >
 | 
			
		||||
      <div className={clsx(classes.Box, whClassStyle)}>
 | 
			
		||||
        <svg width="13" height="4" viewBox="0 0 13 4" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
          <rect width="13" height="4" rx="2" className={whClassStyle} fill="currentColor" />
 | 
			
		||||
        <svg width="13" height="8" viewBox="0 0 13 8" xmlns="http://www.w3.org/2000/svg">
 | 
			
		||||
          <rect y="1" width="13" height="4" rx="2" className={whClassStyle} fill="currentColor" />
 | 
			
		||||
          {isEOL && <rect x="4" width="5" height="6" rx="1" className={clsx(classes.Eol)} fill="#a153ac" />}
 | 
			
		||||
        </svg>
 | 
			
		||||
      </div>
 | 
			
		||||
    </WdTooltipWrapper>
 | 
			
		||||
 
 | 
			
		||||
@@ -16,7 +16,7 @@ export enum SOLAR_SYSTEM_CLASS_IDS {
 | 
			
		||||
  thera = 12,
 | 
			
		||||
  c13 = 13,
 | 
			
		||||
  sentinel = 14,
 | 
			
		||||
  baribican = 15,
 | 
			
		||||
  barbican = 15,
 | 
			
		||||
  vidette = 16,
 | 
			
		||||
  conflux = 17,
 | 
			
		||||
  redoubt = 18,
 | 
			
		||||
@@ -82,7 +82,7 @@ export const SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS = {
 | 
			
		||||
  thera: SOLAR_SYSTEM_CLASS_GROUPS.thera,
 | 
			
		||||
  c13: SOLAR_SYSTEM_CLASS_GROUPS.c13,
 | 
			
		||||
  sentinel: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
 | 
			
		||||
  baribican: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
 | 
			
		||||
  barbican: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
 | 
			
		||||
  vidette: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
 | 
			
		||||
  conflux: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
 | 
			
		||||
  redoubt: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
 | 
			
		||||
@@ -217,7 +217,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
 | 
			
		||||
    wormholeClassID: 14,
 | 
			
		||||
    effectPower: 2,
 | 
			
		||||
    title: 'Class 14 (Sentinel Drifter)',
 | 
			
		||||
    shortTitle: 'Sentinel',
 | 
			
		||||
    shortTitle: 'Sentinel MZ',
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    id: 'barbican',
 | 
			
		||||
@@ -225,7 +225,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
 | 
			
		||||
    wormholeClassID: 15,
 | 
			
		||||
    effectPower: 2,
 | 
			
		||||
    title: 'Class 15 (Barbican Drifter)',
 | 
			
		||||
    shortTitle: 'Barbican',
 | 
			
		||||
    shortTitle: 'Liberated Barbican',
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    id: 'vidette',
 | 
			
		||||
@@ -233,7 +233,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
 | 
			
		||||
    wormholeClassID: 16,
 | 
			
		||||
    effectPower: 2,
 | 
			
		||||
    title: 'Class 16 (Vidette Drifter)',
 | 
			
		||||
    shortTitle: 'Vidette',
 | 
			
		||||
    shortTitle: 'Sanctified Vidette',
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    id: 'conflux',
 | 
			
		||||
@@ -241,7 +241,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
 | 
			
		||||
    wormholeClassID: 17,
 | 
			
		||||
    effectPower: 2,
 | 
			
		||||
    title: 'Class 17 (Conflux Drifter)',
 | 
			
		||||
    shortTitle: 'Conflux',
 | 
			
		||||
    shortTitle: 'Conflux Eyrie',
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    id: 'redoubt',
 | 
			
		||||
@@ -249,7 +249,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
 | 
			
		||||
    wormholeClassID: 18,
 | 
			
		||||
    effectPower: 2,
 | 
			
		||||
    title: 'Class 18 (Redoubt Drifter)',
 | 
			
		||||
    shortTitle: 'Redoubt',
 | 
			
		||||
    shortTitle: 'Azdaja Redoubt',
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    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> =
 | 
			
		||||
  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 = {
 | 
			
		||||
//   ccp1 =  ,
 | 
			
		||||
//   c1 = ,
 | 
			
		||||
@@ -650,6 +653,7 @@ export enum LABELS {
 | 
			
		||||
  l3 = '3',
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
 | 
			
		||||
export const LABELS_INFO: Record<string, any> = {
 | 
			
		||||
  [LABELS.clear]: { id: 'clear', name: 'Clear', shortName: '', icon: '' },
 | 
			
		||||
  [LABELS.la]: { id: 'la', name: 'Label A', shortName: 'A', icon: '' },
 | 
			
		||||
@@ -750,6 +754,17 @@ export const SHIP_SIZES_SIZE = {
 | 
			
		||||
  [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 = {
 | 
			
		||||
  [ShipSizeStatus.small]: 'Frigate wormhole - up to Destroyer | 5K t.',
 | 
			
		||||
  [ShipSizeStatus.medium]: 'Cruise wormhole - up to Battlecruiser | 62K t.',
 | 
			
		||||
 
 | 
			
		||||
@@ -10,5 +10,6 @@ export const convertSystem2Node = (sys: SolarSystemRawType): Node => {
 | 
			
		||||
    position: sys.position,
 | 
			
		||||
    data: sys,
 | 
			
		||||
    draggable: !sys.locked,
 | 
			
		||||
    deletable: !sys.locked,
 | 
			
		||||
  };
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,32 @@
 | 
			
		||||
import { SolarSystemNodeDefault, SolarSystemNodeTheme } from '../components/SolarSystemNode';
 | 
			
		||||
import type { NodeProps } from 'reactflow';
 | 
			
		||||
import type { ComponentType } from 'react';
 | 
			
		||||
import { MapSolarSystemType } from '../map.types';
 | 
			
		||||
import { ConnectionMode } from 'reactflow';
 | 
			
		||||
 | 
			
		||||
export type SolarSystemNodeComponent = ComponentType<NodeProps<MapSolarSystemType>>;
 | 
			
		||||
 | 
			
		||||
interface ThemeBehavior {
 | 
			
		||||
  isPanAndDrag: boolean;
 | 
			
		||||
  nodeComponent: SolarSystemNodeComponent;
 | 
			
		||||
  connectionMode: ConnectionMode;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const THEME_BEHAVIORS: {
 | 
			
		||||
  [key: string]: ThemeBehavior;
 | 
			
		||||
} = {
 | 
			
		||||
  default: {
 | 
			
		||||
    isPanAndDrag: false,
 | 
			
		||||
    nodeComponent: SolarSystemNodeDefault,
 | 
			
		||||
    connectionMode: ConnectionMode.Loose,
 | 
			
		||||
  },
 | 
			
		||||
  pathfinder: {
 | 
			
		||||
    isPanAndDrag: true,
 | 
			
		||||
    nodeComponent: SolarSystemNodeTheme,
 | 
			
		||||
    connectionMode: ConnectionMode.Loose,
 | 
			
		||||
  },
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export function getBehaviorForTheme(themeName: string) {
 | 
			
		||||
  return THEME_BEHAVIORS[themeName] ?? THEME_BEHAVIORS.default;
 | 
			
		||||
}
 | 
			
		||||
@@ -10,7 +10,7 @@ export const isWormholeSpace = (wormholeClassID: number) => {
 | 
			
		||||
    case SOLAR_SYSTEM_CLASS_IDS.c6:
 | 
			
		||||
    case SOLAR_SYSTEM_CLASS_IDS.c13:
 | 
			
		||||
    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.conflux:
 | 
			
		||||
    case SOLAR_SYSTEM_CLASS_IDS.redoubt:
 | 
			
		||||
 
 | 
			
		||||
@@ -2,10 +2,13 @@ import { Node, useReactFlow } from 'reactflow';
 | 
			
		||||
import { useCallback, useRef } from 'react';
 | 
			
		||||
import { CommandAddSystems } from '@/hooks/Mapper/types/mapHandlers.ts';
 | 
			
		||||
import { convertSystem2Node } from '../../helpers';
 | 
			
		||||
import { useLoadSystemStatic } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
 | 
			
		||||
 | 
			
		||||
export const useMapAddSystems = () => {
 | 
			
		||||
  const rf = useReactFlow();
 | 
			
		||||
 | 
			
		||||
  const { addSystemStatic } = useLoadSystemStatic({ systems: [] });
 | 
			
		||||
 | 
			
		||||
  const ref = useRef({ rf });
 | 
			
		||||
  ref.current = { rf };
 | 
			
		||||
 | 
			
		||||
@@ -13,7 +16,10 @@ export const useMapAddSystems = () => {
 | 
			
		||||
    const { rf } = ref.current;
 | 
			
		||||
    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);
 | 
			
		||||
  }, []);
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
import { MapData, useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
 | 
			
		||||
import { useCallback, useRef } from 'react';
 | 
			
		||||
import { CommandKillsUpdated, CommandMapUpdated } from '@/hooks/Mapper/types';
 | 
			
		||||
import { useCallback, useRef } from 'react';
 | 
			
		||||
 | 
			
		||||
export const useMapCommands = () => {
 | 
			
		||||
  const { update } = useMapState();
 | 
			
		||||
@@ -8,13 +8,21 @@ export const useMapCommands = () => {
 | 
			
		||||
  const ref = useRef({ update });
 | 
			
		||||
  ref.current = { update };
 | 
			
		||||
 | 
			
		||||
  const mapUpdated = useCallback(({ hubs }: CommandMapUpdated) => {
 | 
			
		||||
  const mapUpdated = useCallback(({ hubs, system_signatures, kills }: CommandMapUpdated) => {
 | 
			
		||||
    const out: Partial<MapData> = {};
 | 
			
		||||
 | 
			
		||||
    if (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);
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -14,6 +14,7 @@ export const useMapInit = () => {
 | 
			
		||||
  return useCallback(
 | 
			
		||||
    ({
 | 
			
		||||
      systems,
 | 
			
		||||
      system_signatures,
 | 
			
		||||
      kills,
 | 
			
		||||
      connections,
 | 
			
		||||
      wormholes,
 | 
			
		||||
@@ -51,6 +52,10 @@ export const useMapInit = () => {
 | 
			
		||||
        updateData.systems = systems;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (system_signatures) {
 | 
			
		||||
        updateData.systemSignatures = system_signatures;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (kills) {
 | 
			
		||||
        updateData.kills = kills.reduce((acc, x) => ({ ...acc, [x.solar_system_id]: x.kills }), {});
 | 
			
		||||
      }
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,7 @@ export const useMapUpdateSystems = () => {
 | 
			
		||||
        return newSystem;
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      update({ systems: out });
 | 
			
		||||
      update({ systems: out }, true);
 | 
			
		||||
    },
 | 
			
		||||
    [rf, update],
 | 
			
		||||
  );
 | 
			
		||||
 
 | 
			
		||||
@@ -1,3 +1,11 @@
 | 
			
		||||
export * from './useMapHandlers';
 | 
			
		||||
export * from './useUpdateNodes';
 | 
			
		||||
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,15 +1,18 @@
 | 
			
		||||
import { useEffect, useState } from 'react';
 | 
			
		||||
import { BackgroundVariant } from 'reactflow';
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
export function useBackgroundVars(themeName?: string) {
 | 
			
		||||
  const [variant, setVariant] = useState<BackgroundVariant>(BackgroundVariant.Dots);
 | 
			
		||||
  const [gap, setGap] = useState<number>(16);
 | 
			
		||||
  const [size, setSize] = useState<number>(1);
 | 
			
		||||
  const [color, setColor] = useState('#81818b')
 | 
			
		||||
  const [color, setColor] = useState('#81818b');
 | 
			
		||||
  const [snapSize, setSnapSize] = useState<number>(25);
 | 
			
		||||
 | 
			
		||||
  useEffect(() => { 
 | 
			
		||||
    let themeEl = document.querySelector('.pathfinder-theme, .neon-theme');
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    // match any element whose entire `class` attribute ends with "-theme"
 | 
			
		||||
    let themeEl = document.querySelector('[class$="-theme"]');
 | 
			
		||||
 | 
			
		||||
    // If none is found, fall back to the <html> element
 | 
			
		||||
    if (!themeEl) {
 | 
			
		||||
      themeEl = document.documentElement;
 | 
			
		||||
    }
 | 
			
		||||
@@ -18,6 +21,7 @@ export function useBackgroundVars(themeName?: string) {
 | 
			
		||||
 | 
			
		||||
    const rawVariant = style.getPropertyValue('--rf-bg-variant').replace(/['"]/g, '').trim().toLowerCase();
 | 
			
		||||
    let finalVariant: BackgroundVariant = BackgroundVariant.Dots;
 | 
			
		||||
 | 
			
		||||
    if (rawVariant === 'lines') {
 | 
			
		||||
      finalVariant = BackgroundVariant.Lines;
 | 
			
		||||
    } else if (rawVariant === 'cross') {
 | 
			
		||||
@@ -26,17 +30,19 @@ export function useBackgroundVars(themeName?: string) {
 | 
			
		||||
 | 
			
		||||
    const cssVarGap = style.getPropertyValue('--rf-bg-gap');
 | 
			
		||||
    const cssVarSize = style.getPropertyValue('--rf-bg-size');
 | 
			
		||||
    const cssVarSnapSize = style.getPropertyValue('--rf-snap-size');
 | 
			
		||||
    const cssColor = style.getPropertyValue('--rf-bg-pattern-color');
 | 
			
		||||
 | 
			
		||||
    const gapNum = parseInt(cssVarGap, 10) || 16;
 | 
			
		||||
    const sizeNum = parseInt(cssVarSize, 10) || 1;
 | 
			
		||||
    const snapSize = parseInt(cssVarSnapSize, 10) || 25; //react-flow default
 | 
			
		||||
 | 
			
		||||
    setVariant(finalVariant);
 | 
			
		||||
    setGap(gapNum);
 | 
			
		||||
    setSize(sizeNum);
 | 
			
		||||
    setColor(cssColor);
 | 
			
		||||
    setSnapSize(snapSize);
 | 
			
		||||
  }, [themeName]);
 | 
			
		||||
 | 
			
		||||
  }, [themeName]); 
 | 
			
		||||
 | 
			
		||||
  return { variant, gap, size, color };
 | 
			
		||||
  return { variant, gap, size, color, snapSize };
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,43 @@
 | 
			
		||||
import { useMemo } from 'react';
 | 
			
		||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
import { useSystemKills } from '@/hooks/Mapper/components/mapInterface/widgets/WSystemKills/hooks/useSystemKills.ts';
 | 
			
		||||
 | 
			
		||||
interface UseKillsCounterProps {
 | 
			
		||||
  realSystemId: string;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function useKillsCounter({ realSystemId }: UseKillsCounterProps) {
 | 
			
		||||
  const { data: mapData, outCommand } = useMapRootState();
 | 
			
		||||
  const { systems } = mapData;
 | 
			
		||||
 | 
			
		||||
  const systemNameMap = useMemo(() => {
 | 
			
		||||
    const m: Record<string, string> = {};
 | 
			
		||||
    systems.forEach(sys => {
 | 
			
		||||
      m[sys.id] = sys.temporary_name || sys.name || '???';
 | 
			
		||||
    });
 | 
			
		||||
    return m;
 | 
			
		||||
  }, [systems]);
 | 
			
		||||
 | 
			
		||||
  const { kills: allKills, isLoading } = useSystemKills({
 | 
			
		||||
    systemId: realSystemId,
 | 
			
		||||
    outCommand,
 | 
			
		||||
    showAllVisible: false,
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  const filteredKills = useMemo(() => {
 | 
			
		||||
    if (!allKills || allKills.length === 0) return [];
 | 
			
		||||
 | 
			
		||||
    // Sort kills by time, most recent first, but don't limit the number of kills
 | 
			
		||||
    return [...allKills].sort((a, b) => {
 | 
			
		||||
      const aTime = a.kill_time ? new Date(a.kill_time).getTime() : 0;
 | 
			
		||||
      const bTime = b.kill_time ? new Date(b.kill_time).getTime() : 0;
 | 
			
		||||
      return bTime - aTime;
 | 
			
		||||
    });
 | 
			
		||||
  }, [allKills]);
 | 
			
		||||
 | 
			
		||||
  return {
 | 
			
		||||
    isLoading,
 | 
			
		||||
    kills: filteredKills,
 | 
			
		||||
    systemNameMap,
 | 
			
		||||
  };
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										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 };
 | 
			
		||||
}
 | 
			
		||||
@@ -70,7 +70,7 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
 | 
			
		||||
              setTimeout(() => addConnections(data as CommandAddConnections), 100);
 | 
			
		||||
              break;
 | 
			
		||||
            case Commands.removeConnections:
 | 
			
		||||
              removeConnections(data as CommandRemoveConnections);
 | 
			
		||||
              setTimeout(() => removeConnections(data as CommandRemoveConnections), 100);
 | 
			
		||||
              break;
 | 
			
		||||
            case Commands.charactersUpdated:
 | 
			
		||||
              charactersUpdated(data as CommandCharactersUpdated);
 | 
			
		||||
@@ -127,6 +127,25 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
 | 
			
		||||
              // do nothing here
 | 
			
		||||
              break;
 | 
			
		||||
 | 
			
		||||
            case Commands.detailedKillsUpdated:
 | 
			
		||||
              // do nothing here
 | 
			
		||||
              break;
 | 
			
		||||
 | 
			
		||||
            case Commands.characterActivityData:
 | 
			
		||||
              break;
 | 
			
		||||
 | 
			
		||||
            case Commands.trackingCharactersData:
 | 
			
		||||
              break;
 | 
			
		||||
 | 
			
		||||
            case Commands.updateActivity:
 | 
			
		||||
              break;
 | 
			
		||||
 | 
			
		||||
            case Commands.updateTracking:
 | 
			
		||||
              break;
 | 
			
		||||
 | 
			
		||||
            case Commands.userSettingsUpdated:
 | 
			
		||||
              break;
 | 
			
		||||
 | 
			
		||||
            default:
 | 
			
		||||
              console.warn(`Map handlers: Unknown command: ${type}`, data);
 | 
			
		||||
              break;
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,40 @@
 | 
			
		||||
import { useEffect, useState, useCallback } from 'react';
 | 
			
		||||
import { useMapEventListener } from '@/hooks/Mapper/events';
 | 
			
		||||
import { Commands } from '@/hooks/Mapper/types';
 | 
			
		||||
 | 
			
		||||
interface Kill {
 | 
			
		||||
  solar_system_id: number | string;
 | 
			
		||||
  kills: number;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
interface MapEvent {
 | 
			
		||||
  name: Commands;
 | 
			
		||||
  data?: any;
 | 
			
		||||
  payload?: Kill[];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export function useNodeKillsCount(systemId: number | string, initialKillsCount: number | null): number | null {
 | 
			
		||||
  const [killsCount, setKillsCount] = useState<number | null>(initialKillsCount);
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    setKillsCount(initialKillsCount);
 | 
			
		||||
  }, [initialKillsCount]);
 | 
			
		||||
 | 
			
		||||
  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') {
 | 
			
		||||
          setKillsCount(killForSystem.kills);
 | 
			
		||||
        }
 | 
			
		||||
        return true;
 | 
			
		||||
      }
 | 
			
		||||
      return false;
 | 
			
		||||
    },
 | 
			
		||||
    [systemId],
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  useMapEventListener(handleEvent);
 | 
			
		||||
 | 
			
		||||
  return killsCount;
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,234 @@
 | 
			
		||||
import { useMemo } from 'react';
 | 
			
		||||
import { MapSolarSystemType } from '../map.types';
 | 
			
		||||
import { NodeProps } from 'reactflow';
 | 
			
		||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
import { useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
 | 
			
		||||
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider';
 | 
			
		||||
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick';
 | 
			
		||||
import { REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
 | 
			
		||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace';
 | 
			
		||||
import { getSystemClassStyles } from '@/hooks/Mapper/components/map/helpers';
 | 
			
		||||
import { sortWHClasses } from '@/hooks/Mapper/helpers';
 | 
			
		||||
import { CharacterTypeRaw, OutCommand, SystemSignature } from '@/hooks/Mapper/types';
 | 
			
		||||
import { useUnsplashedSignatures } from './useUnsplashedSignatures';
 | 
			
		||||
import { useSystemName } from './useSystemName';
 | 
			
		||||
import { LabelInfo, useLabelsInfo } from './useLabelsInfo';
 | 
			
		||||
import { getSystemStaticInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
 | 
			
		||||
 | 
			
		||||
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',
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
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 {
 | 
			
		||||
    id: solar_system_id,
 | 
			
		||||
    locked,
 | 
			
		||||
    name,
 | 
			
		||||
    tag,
 | 
			
		||||
    status,
 | 
			
		||||
    labels,
 | 
			
		||||
    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,
 | 
			
		||||
  } = 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,
 | 
			
		||||
      kills,
 | 
			
		||||
      userCharacters,
 | 
			
		||||
      isConnecting,
 | 
			
		||||
      hoverNodeId,
 | 
			
		||||
      visibleNodes,
 | 
			
		||||
      showKSpaceBG,
 | 
			
		||||
      isThickConnections,
 | 
			
		||||
    },
 | 
			
		||||
    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,
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  const killsCount = useMemo(() => kills[parseInt(solar_system_id)] ?? null, [kills, solar_system_id]);
 | 
			
		||||
  const killsActivityType = killsCount ? getActivityType(killsCount) : null;
 | 
			
		||||
 | 
			
		||||
  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 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,
 | 
			
		||||
    solarSystemId: solar_system_id.toString(),
 | 
			
		||||
    locked,
 | 
			
		||||
    hubs: hubsAsStrings,
 | 
			
		||||
    name,
 | 
			
		||||
    isConnecting,
 | 
			
		||||
    hoverNodeId,
 | 
			
		||||
    charactersInSystem,
 | 
			
		||||
    unsplashedLeft,
 | 
			
		||||
    unsplashedRight,
 | 
			
		||||
    isThickConnections,
 | 
			
		||||
    classTitle: class_title,
 | 
			
		||||
    temporaryName: computedTemporaryName,
 | 
			
		||||
    regionName: region_name,
 | 
			
		||||
    solarSystemName: solar_system_name,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  return nodeVars;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export interface SolarSystemNodeVars {
 | 
			
		||||
  id: string;
 | 
			
		||||
  selected: boolean;
 | 
			
		||||
  visible: boolean;
 | 
			
		||||
  isWormhole: boolean;
 | 
			
		||||
  classTitleColor: string | null;
 | 
			
		||||
  killsCount: number | null;
 | 
			
		||||
  killsActivityType: string | null;
 | 
			
		||||
  hasUserCharacters: boolean;
 | 
			
		||||
  showHandlers: boolean;
 | 
			
		||||
  regionClass: string | null;
 | 
			
		||||
  systemName: string;
 | 
			
		||||
  customName?: string | null;
 | 
			
		||||
  labelCustom: string | null;
 | 
			
		||||
  isShattered: boolean;
 | 
			
		||||
  tag?: string | null;
 | 
			
		||||
  status?: number;
 | 
			
		||||
  labelsInfo: LabelInfo[];
 | 
			
		||||
  dbClick: (event: React.MouseEvent<HTMLDivElement>) => void;
 | 
			
		||||
  sortedStatics: Array<string | number>;
 | 
			
		||||
  effectName: string | null;
 | 
			
		||||
  regionName: string | null;
 | 
			
		||||
  solarSystemId: string;
 | 
			
		||||
  solarSystemName: string | null;
 | 
			
		||||
  locked: boolean;
 | 
			
		||||
  hubs: string[];
 | 
			
		||||
  name: string | null;
 | 
			
		||||
  isConnecting: boolean;
 | 
			
		||||
  hoverNodeId: string | null;
 | 
			
		||||
  charactersInSystem: Array<CharacterTypeRaw>;
 | 
			
		||||
  userCharacters: string[];
 | 
			
		||||
  unsplashedLeft: Array<SystemSignature>;
 | 
			
		||||
  unsplashedRight: Array<SystemSignature>;
 | 
			
		||||
  isThickConnections: boolean;
 | 
			
		||||
  classTitle: string | null;
 | 
			
		||||
  temporaryName?: string | null;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										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 { useMemo } from 'react';
 | 
			
		||||
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
 | 
			
		||||
 | 
			
		||||
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]);
 | 
			
		||||
}
 | 
			
		||||
@@ -6,9 +6,9 @@ import { SolarSystemRawType } from '@/hooks/Mapper/types';
 | 
			
		||||
const useThrottle = () => {
 | 
			
		||||
  const throttleSeed = useRef<number | null>(null);
 | 
			
		||||
 | 
			
		||||
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
 | 
			
		||||
  const throttleFunction = useRef((func: any, delay = 200) => {
 | 
			
		||||
    if (!throttleSeed.current) {
 | 
			
		||||
      // Call the callback immediately for the first time
 | 
			
		||||
      func();
 | 
			
		||||
      throttleSeed.current = setTimeout(() => {
 | 
			
		||||
        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));
 | 
			
		||||
    update({ visibleNodes });
 | 
			
		||||
  }, [nodes]);
 | 
			
		||||
  }, [getViewport, nodes, update]);
 | 
			
		||||
 | 
			
		||||
  useOnViewportChange({
 | 
			
		||||
    onChange: () => throttle(updateNodesVisibility.bind(this)),
 | 
			
		||||
@@ -84,5 +84,5 @@ export const useUpdateNodes = (nodes: Node<SolarSystemRawType>[]) => {
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    updateNodesVisibility();
 | 
			
		||||
  }, [nodes]);
 | 
			
		||||
  }, [nodes, updateNodesVisibility]);
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,36 @@
 | 
			
		||||
@import './eve-common-variables';
 | 
			
		||||
@import './eve-common';
 | 
			
		||||
 | 
			
		||||
.default-theme {
 | 
			
		||||
  --rf-bg-color: #0C0A09;
 | 
			
		||||
  --rf-soft-bg-color: #171717;
 | 
			
		||||
 | 
			
		||||
  --rf-node-bg-color: #202020;
 | 
			
		||||
  --rf-node-soft-bg-color: #202020;
 | 
			
		||||
  --rf-text-color: #ffffff;
 | 
			
		||||
  --rf-tag-color:  #38BDF8;
 | 
			
		||||
  --rf-region-name: #D6D3D1;
 | 
			
		||||
  --rf-custom-name: #93C5FD;
 | 
			
		||||
  --rf-node-font-family: 'Shentox', 'Rogan', sans-serif !important;
 | 
			
		||||
  --rf-node-font-weight: 500;
 | 
			
		||||
 | 
			
		||||
  --rf-bg-variant: "dots";
 | 
			
		||||
  --rf-bg-gap: 15;
 | 
			
		||||
  --rf-bg-size: 1;
 | 
			
		||||
  --rf-bg-pattern-color: #81818a;
 | 
			
		||||
 | 
			
		||||
  --pastel-blue: #5a7d9a;
 | 
			
		||||
  --pastel-pink: #d291bc;
 | 
			
		||||
  --pastel-green: #88b04b;
 | 
			
		||||
  --pastel-yellow: #ffdd59;
 | 
			
		||||
 | 
			
		||||
  --dark-bg: #2d2d2d;
 | 
			
		||||
  --text-color: #ffffff;
 | 
			
		||||
  --tooltip-bg: #202020;
 | 
			
		||||
 | 
			
		||||
  --window-corner: #72716f;
 | 
			
		||||
 | 
			
		||||
  --rf-local-counter-font-weight: 500;
 | 
			
		||||
  --rf-node-local-counter: inherit;
 | 
			
		||||
  --rf-has-user-characters: #ffc75d;
 | 
			
		||||
}
 | 
			
		||||
@@ -1,114 +1,122 @@
 | 
			
		||||
 | 
			
		||||
$friendlyBase: #3bbd39;
 | 
			
		||||
$friendlyAlpha: #3bbd3952;
 | 
			
		||||
$friendlyDark20: darken($friendlyBase, 20%);
 | 
			
		||||
$friendlyDark30: darken($friendlyBase, 30%);
 | 
			
		||||
$friendlyDark5:  darken($friendlyBase, 5%);
 | 
			
		||||
 | 
			
		||||
$lookingForBase: #43c2fd;
 | 
			
		||||
$lookingForAlpha: rgba(67, 176, 253, 0.48);
 | 
			
		||||
$lookingForDark15: darken($lookingForBase, 15%);
 | 
			
		||||
 | 
			
		||||
$homeBase: rgb(179, 253, 67);
 | 
			
		||||
$homeAlpha: rgba(186, 248, 48, 0.32);
 | 
			
		||||
$homeBackground: #a0fa5636;
 | 
			
		||||
$homeDark30: darken($homeBase, 30%);
 | 
			
		||||
 | 
			
		||||
:root {
 | 
			
		||||
    --pastel-blue: #5a7d9a;
 | 
			
		||||
    --pastel-pink: #d291bc;
 | 
			
		||||
    --pastel-green: #88b04b;
 | 
			
		||||
    --pastel-yellow: #ffdd59;
 | 
			
		||||
    --dark-bg: #2d2d2d;
 | 
			
		||||
    --text-color: #ffffff;
 | 
			
		||||
    --tooltip-bg: #202020;
 | 
			
		||||
  --pastel-blue: #5a7d9a;
 | 
			
		||||
  --pastel-pink: #d291bc;
 | 
			
		||||
  --pastel-green: #88b04b;
 | 
			
		||||
  --pastel-yellow: #ffdd59;
 | 
			
		||||
  --dark-bg: #2d2d2d;
 | 
			
		||||
  --text-color: #ffffff;
 | 
			
		||||
  --tooltip-bg: #202020;
 | 
			
		||||
 | 
			
		||||
    --pastel-blue-darken10: #4f6b86;
 | 
			
		||||
    --pastel-blue-lighten10: #6da3af;
 | 
			
		||||
    --pastel-pink-darken10: #bb7ca9;
 | 
			
		||||
    --pastel-pink-lighten10: #e0a6cb;
 | 
			
		||||
  --pastel-blue-darken10: #4f6b86;
 | 
			
		||||
  --pastel-blue-lighten10: #6da3af;
 | 
			
		||||
  --pastel-pink-darken10: #bb7ca9;
 | 
			
		||||
  --pastel-pink-lighten10: #e0a6cb;
 | 
			
		||||
  --pastel-green-darken10: #79a244;
 | 
			
		||||
  --pastel-green-lighten10: #99cf52;
 | 
			
		||||
  --pastel-yellow-darken10: #e6c44f;
 | 
			
		||||
  --pastel-yellow-lighten10: #ffe874;
 | 
			
		||||
 | 
			
		||||
    --pastel-green-darken10: #79a244;
 | 
			
		||||
    --pastel-green-lighten10: #99cf52;
 | 
			
		||||
  --eve-link-color-default: #333;
 | 
			
		||||
  --eve-link-color-top-mass-0: #333;
 | 
			
		||||
  --eve-link-color-top-mass-1: #5a4520;
 | 
			
		||||
  --eve-link-color-top-mass-2: #672c2c;
 | 
			
		||||
  --eve-link-color-middle-mass-0: #333;
 | 
			
		||||
  --eve-link-color-middle-mass-1: #333;
 | 
			
		||||
  --eve-link-color-middle-mass-2: #333;
 | 
			
		||||
  --eve-link-color-middle-time-0: #5c5c5c;
 | 
			
		||||
  --eve-link-color-middle-time-1: #ff00cd;
 | 
			
		||||
  --eve-link-color-middle-time-1-border: #99f3ff;
 | 
			
		||||
  --eve-link-color-top-mass-1-time-1: #796300;
 | 
			
		||||
  --eve-link-color-top-mass-2-time-1: #8c1717;
 | 
			
		||||
  --eve-link-color-temp: orange;
 | 
			
		||||
 | 
			
		||||
    --pastel-yellow-darken10: #e6c44f;
 | 
			
		||||
    --pastel-yellow-lighten10: #ffe874;
 | 
			
		||||
  --eve-effect-pulsar: #40aef5;
 | 
			
		||||
  --eve-effect-magnetar: #f058f8;
 | 
			
		||||
  --eve-effect-wolfRayet: #ef7843;
 | 
			
		||||
  --eve-effect-blackHole: #1b1b1b;
 | 
			
		||||
  --eve-effect-cataclysmicVariable: #ffea90;
 | 
			
		||||
  --eve-effect-redGiant: #fd3c3c;
 | 
			
		||||
  --eve-effect-dazhLiminalityLocus: #ff6464;
 | 
			
		||||
  --eve-effect-imperialStellarObservatory: #6991ce;
 | 
			
		||||
  --eve-effect-stateStellarObservatory: #6991ce;
 | 
			
		||||
  --eve-effect-republicStellarObservatory: #6991ce;
 | 
			
		||||
  --eve-effect-federalStellarObservatory: #6991ce;
 | 
			
		||||
 | 
			
		||||
    /* Eve Link Colors */
 | 
			
		||||
    --eve-link-color-default: #333;
 | 
			
		||||
    --eve-link-color-top-mass-0: #333;
 | 
			
		||||
    --eve-link-color-top-mass-1: #5a4520;
 | 
			
		||||
    --eve-link-color-top-mass-2: #672c2c;
 | 
			
		||||
    --eve-link-color-middle-mass-0: #333;
 | 
			
		||||
    --eve-link-color-middle-mass-1: #333;
 | 
			
		||||
    --eve-link-color-middle-mass-2: #333;
 | 
			
		||||
    --eve-link-color-middle-time-0: #5c5c5c;
 | 
			
		||||
    --eve-link-color-middle-time-1: #ff00cd;
 | 
			
		||||
    --eve-link-color-middle-time-1-border: #99f3ff;
 | 
			
		||||
    --eve-link-color-top-mass-1-time-1: #796300;
 | 
			
		||||
    --eve-link-color-top-mass-2-time-1: #8c1717;
 | 
			
		||||
    --eve-link-color-temp: orange;
 | 
			
		||||
  --eve-wh-type-color-high: #5dffd2;
 | 
			
		||||
  --eve-wh-type-color-low: #f79400;
 | 
			
		||||
  --eve-wh-type-color-null: #fc3c3c;
 | 
			
		||||
  --eve-wh-type-color-c1: #69bfce;
 | 
			
		||||
  --eve-wh-type-color-c2: #6991ce;
 | 
			
		||||
  --eve-wh-type-color-c3: #a8cb70;
 | 
			
		||||
  --eve-wh-type-color-c4: #e39c68;
 | 
			
		||||
  --eve-wh-type-color-c5: #de8686;
 | 
			
		||||
  --eve-wh-type-color-c6: #e76363;
 | 
			
		||||
  --eve-wh-type-color-c13: #988cb5;
 | 
			
		||||
  --eve-wh-type-color-drifter: #ff44f6;
 | 
			
		||||
  --eve-wh-type-color-thera: #ffffff;
 | 
			
		||||
  --eve-wh-type-color-zarzakh: #212121;
 | 
			
		||||
 | 
			
		||||
    /* Wormhole Effects */
 | 
			
		||||
    --eve-effect-pulsar: #40aef5;
 | 
			
		||||
    --eve-effect-magnetar: #f058f8;
 | 
			
		||||
    --eve-effect-wolfRayet: #ef7843;
 | 
			
		||||
    --eve-effect-blackHole: #1b1b1b;
 | 
			
		||||
    --eve-effect-cataclysmicVariable: #ffea90;
 | 
			
		||||
    --eve-effect-redGiant: #fd3c3c;
 | 
			
		||||
    --eve-effect-dazhLiminalityLocus: #ff6464;
 | 
			
		||||
    --eve-effect-imperialStellarObservatory: #6991ce;
 | 
			
		||||
    --eve-effect-stateStellarObservatory: #6991ce;
 | 
			
		||||
    --eve-effect-republicStellarObservatory: #6991ce;
 | 
			
		||||
    --eve-effect-federalStellarObservatory: #6991ce;
 | 
			
		||||
  --eve-security-color-10: #2c74df;
 | 
			
		||||
  --eve-security-color-09: #3998e8;
 | 
			
		||||
  --eve-security-color-08: #4dcbf5;
 | 
			
		||||
  --eve-security-color-07: #60d8a2;
 | 
			
		||||
  --eve-security-color-06: #71e454;
 | 
			
		||||
  --eve-security-color-05: #f2fc81;
 | 
			
		||||
  --eve-security-color-04: #d96c07;
 | 
			
		||||
  --eve-security-color-03: #cb440f;
 | 
			
		||||
  --eve-security-color-02: #b91117;
 | 
			
		||||
  --eve-security-color-01: #732020;
 | 
			
		||||
  --eve-security-color-00: #8b3263;
 | 
			
		||||
  --eve-security-color-m-01: #8b3263;
 | 
			
		||||
  --eve-security-color-m-02: #8b3263;
 | 
			
		||||
  --eve-security-color-m-03: #8b3263;
 | 
			
		||||
  --eve-security-color-m-04: #8b3263;
 | 
			
		||||
  --eve-security-color-m-05: #8b3263;
 | 
			
		||||
  --eve-security-color-m-06: #8b3263;
 | 
			
		||||
  --eve-security-color-m-07: #8b3263;
 | 
			
		||||
  --eve-security-color-m-08: #8b3263;
 | 
			
		||||
  --eve-security-color-m-09: #8b3263;
 | 
			
		||||
  --eve-security-color-m-10: #8b3263;
 | 
			
		||||
 | 
			
		||||
    /* WH Types */
 | 
			
		||||
    --eve-wh-type-color-high: #5dffd2;
 | 
			
		||||
    --eve-wh-type-color-low: #f79400;
 | 
			
		||||
    --eve-wh-type-color-null: #fc3c3c;
 | 
			
		||||
    --eve-wh-type-color-c1: #69bfce;
 | 
			
		||||
    --eve-wh-type-color-c2: #6991ce;
 | 
			
		||||
    --eve-wh-type-color-c3: #a8cb70;
 | 
			
		||||
    --eve-wh-type-color-c4: #e39c68;
 | 
			
		||||
    --eve-wh-type-color-c5: #de8686;
 | 
			
		||||
    --eve-wh-type-color-c6: #e76363;
 | 
			
		||||
    --eve-wh-type-color-c13: #988cb5;
 | 
			
		||||
    --eve-wh-type-color-drifter: #ff44f6;
 | 
			
		||||
    --eve-wh-type-color-thera: #ffffff;
 | 
			
		||||
    --eve-wh-type-color-zarzakh: #212121;
 | 
			
		||||
  --eve-solar-system-status-unknown: transparent;
 | 
			
		||||
  --eve-solar-system-status-color-unknown: transparent;
 | 
			
		||||
  --eve-solar-system-status-home:                #{$homeAlpha};
 | 
			
		||||
  --eve-solar-system-status-color-home:          #{$homeBase};
 | 
			
		||||
  --eve-solar-system-status-color-background:    #{$homeBackground};
 | 
			
		||||
  --eve-solar-system-status-color-home-dark30:   #{$homeDark30};
 | 
			
		||||
  --eve-solar-system-status-friendly:            #{$friendlyAlpha};
 | 
			
		||||
  --eve-solar-system-status-color-friendly:      #{$friendlyBase};
 | 
			
		||||
  --eve-solar-system-status-friendly-dark30:     #{$friendlyDark30};
 | 
			
		||||
  --eve-solar-system-status-color-friendly-dark20: #{$friendlyDark20};
 | 
			
		||||
  --eve-solar-system-status-color-friendly-dark5:  #{$friendlyDark5};
 | 
			
		||||
  --eve-solar-system-status-lookingFor:              #{$lookingForAlpha};
 | 
			
		||||
  --eve-solar-system-status-color-lookingFor:        #{$lookingForBase};
 | 
			
		||||
  --eve-solar-system-status-color-lookingFor-dark15: #{$lookingForDark15};
 | 
			
		||||
  --eve-solar-system-status-warning: #906518a6;
 | 
			
		||||
  --eve-solar-system-status-color-warning: #ffb93b;
 | 
			
		||||
  --eve-solar-system-status-target: #b439ff6b;
 | 
			
		||||
  --eve-solar-system-status-color-target: #b439ff;
 | 
			
		||||
  --eve-solar-system-status-dangerous: #d54040;
 | 
			
		||||
  --eve-solar-system-status-color-dangerous: #d54040;
 | 
			
		||||
 | 
			
		||||
    /* Security Colors */
 | 
			
		||||
    --eve-security-color-10: #2c74df;
 | 
			
		||||
    --eve-security-color-09: #3998e8;
 | 
			
		||||
    --eve-security-color-08: #4dcbf5;
 | 
			
		||||
    --eve-security-color-07: #60d8a2;
 | 
			
		||||
    --eve-security-color-06: #71e454;
 | 
			
		||||
    --eve-security-color-05: #f2fc81;
 | 
			
		||||
    --eve-security-color-04: #d96c07;
 | 
			
		||||
    --eve-security-color-03: #cb440f;
 | 
			
		||||
    --eve-security-color-02: #b91117;
 | 
			
		||||
    --eve-security-color-01: #732020;
 | 
			
		||||
    --eve-security-color-00: #8b3263;
 | 
			
		||||
    --eve-security-color-m-01: #8b3263;
 | 
			
		||||
    --eve-security-color-m-02: #8b3263;
 | 
			
		||||
    --eve-security-color-m-03: #8b3263;
 | 
			
		||||
    --eve-security-color-m-04: #8b3263;
 | 
			
		||||
    --eve-security-color-m-05: #8b3263;
 | 
			
		||||
    --eve-security-color-m-06: #8b3263;
 | 
			
		||||
    --eve-security-color-m-07: #8b3263;
 | 
			
		||||
    --eve-security-color-m-08: #8b3263;
 | 
			
		||||
    --eve-security-color-m-09: #8b3263;
 | 
			
		||||
    --eve-security-color-m-10: #8b3263;
 | 
			
		||||
 | 
			
		||||
    /* Solar System Status */
 | 
			
		||||
    --eve-solar-system-status-unknown: transparent;
 | 
			
		||||
    --eve-solar-system-status-friendly: #3bbd3952;
 | 
			
		||||
    --eve-solar-system-status-warning: #906518a6;
 | 
			
		||||
    --eve-solar-system-status-target: #b439ff6b;
 | 
			
		||||
    --eve-solar-system-status-dangerous: #d54040;
 | 
			
		||||
    --eve-solar-system-status-lookingFor: rgba(67, 176, 253, 0.48);
 | 
			
		||||
    --eve-solar-system-status-home: rgb(197, 253, 67);
 | 
			
		||||
 | 
			
		||||
    --eve-solar-system-status-color-unknown: transparent;
 | 
			
		||||
    --eve-solar-system-status-color-friendly: #3bbd39;
 | 
			
		||||
    --eve-solar-system-status-color-warning: #ffb93b;
 | 
			
		||||
    --eve-solar-system-status-color-target: #b439ff;
 | 
			
		||||
    --eve-solar-system-status-color-dangerous: #d54040;
 | 
			
		||||
    --eve-solar-system-status-color-lookingFor: #43c2fd;
 | 
			
		||||
    --eve-solar-system-status-color-home: rgb(197, 253, 67);
 | 
			
		||||
 | 
			
		||||
    --eve-solar-system-status-color-friendly-dark20: #2d9b2e;
 | 
			
		||||
    --eve-solar-system-status-friendly-dark30: #28892a;
 | 
			
		||||
    --eve-solar-system-status-color-friendly-dark5: #38b538;
 | 
			
		||||
    --eve-solar-system-status-color-lookingFor-dark15: #32aadf;
 | 
			
		||||
 | 
			
		||||
    /* Context Menu */
 | 
			
		||||
    --conn-time-eol: #7452c3e3;
 | 
			
		||||
    --conn-frigate: #325d88;
 | 
			
		||||
    --conn-save: rgba(155, 102, 45, 0.85);
 | 
			
		||||
    --selected-item-bg: rgba(98, 98, 98, 0.33);
 | 
			
		||||
  }
 | 
			
		||||
  --conn-time-eol: #7452c3e3;
 | 
			
		||||
  --conn-frigate: #325d88;
 | 
			
		||||
  --conn-save: rgba(155, 102, 45, 0.85);
 | 
			
		||||
  --selected-item-bg: rgba(98, 98, 98, 0.33);
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -429,7 +429,63 @@
 | 
			
		||||
  color: var(--eve-solar-system-status-color-dangerous);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Shapes */
 | 
			
		||||
.eve-system-status-clear {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-unknown);
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-home {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-home);
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-friendly {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-friendly);
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-lookingFor {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-lookingFor);
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-warning {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-warning);
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-target {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-target);
 | 
			
		||||
}
 | 
			
		||||
.eve-system-status-dangerous {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-dangerous);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.eve-system-status-clear {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-unknown);
 | 
			
		||||
  color: var(--eve-solar-system-status-color-unknown);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.eve-system-status-home {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-home);
 | 
			
		||||
  color: var(--eve-solar-system-status-color-home);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.eve-system-status-friendly {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-friendly);
 | 
			
		||||
  color: var(--eve-solar-system-status-color-friendly);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.eve-system-status-lookingFor {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-lookingFor);
 | 
			
		||||
  color: var(--eve-solar-system-status-color-lookingFor);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.eve-system-status-warning {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-warning);
 | 
			
		||||
  color: var(--eve-solar-system-status-color-warning);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.eve-system-status-target {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-target);
 | 
			
		||||
  color: var(--eve-solar-system-status-color-target);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.eve-system-status-dangerous {
 | 
			
		||||
  background-color: var(--eve-solar-system-status-dangerous);
 | 
			
		||||
  color: var(--eve-solar-system-status-color-dangerous);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.wd-route-system-shape-triangle {
 | 
			
		||||
  clip-path: polygon(50% 0, 0 100%, 100% 100%);
 | 
			
		||||
}
 | 
			
		||||
@@ -485,3 +541,33 @@
 | 
			
		||||
.wd-marker-bookmark-color-danger {
 | 
			
		||||
  background-color: #d10600;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
.react-flow__minimap-node {
 | 
			
		||||
  fill: #ffb03a;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.react-flow__minimap {
 | 
			
		||||
  border: 1px solid #282828;
 | 
			
		||||
  border-radius: 4px;
 | 
			
		||||
  background-color: rgb(47 37 37) !important;
 | 
			
		||||
  overflow: hidden;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.react-flow__minimap-mask {
 | 
			
		||||
  stroke-width: 2px;
 | 
			
		||||
  fill: rgba(0, 0, 0, 0.5);
 | 
			
		||||
  mix-blend-mode: overlay;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.react-flow__minimap-mask {
 | 
			
		||||
  stroke-width: 2px;
 | 
			
		||||
  fill: rgb(0 0 0 / 50%) !important;
 | 
			
		||||
  mix-blend-mode: inherit;
 | 
			
		||||
  opacity: 1;
 | 
			
		||||
  stroke: #fff;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.context-menu-active {
 | 
			
		||||
  background-color: rgba(131, 131, 131, 0.33);
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,2 +1,2 @@
 | 
			
		||||
@import './neon-theme.scss'; 
 | 
			
		||||
@import './default-theme.scss'; 
 | 
			
		||||
@import './pathfinder-theme.scss'; 
 | 
			
		||||
@@ -1,72 +0,0 @@
 | 
			
		||||
@import './eve-common-variables';
 | 
			
		||||
@import './eve-common';
 | 
			
		||||
 | 
			
		||||
.neon-theme {
 | 
			
		||||
  --rf-bg-color: #000000;
 | 
			
		||||
  --rf-soft-bg-color: #171717;
 | 
			
		||||
 | 
			
		||||
  --rf-node-bg-color: #202020;
 | 
			
		||||
  --rf-node-soft-bg-color: #2b2b2b;
 | 
			
		||||
  --rf-text-color: #ffffff;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
  --rf-bg-variant: "dots";
 | 
			
		||||
  --rf-bg-gap: 16;
 | 
			
		||||
  --rf-bg-size: 1;
 | 
			
		||||
  --rf-bg-pattern-color: #81818a;
 | 
			
		||||
 | 
			
		||||
  --pastel-blue: #5a7d9a;
 | 
			
		||||
  --pastel-pink: #d291bc;
 | 
			
		||||
  --pastel-green: #88b04b;
 | 
			
		||||
  --pastel-yellow: #ffdd59;
 | 
			
		||||
 | 
			
		||||
  --dark-bg: #2d2d2d;
 | 
			
		||||
  --text-color: #ffffff;
 | 
			
		||||
  --tooltip-bg: #202020;
 | 
			
		||||
 | 
			
		||||
  .react-flow {
 | 
			
		||||
    color: var(--text-color);
 | 
			
		||||
 | 
			
		||||
    &__pane {
 | 
			
		||||
      cursor: auto;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    &__minimap {
 | 
			
		||||
      background-color: rgba(66, 66, 66, 1);
 | 
			
		||||
      opacity: 0.7;
 | 
			
		||||
      border: 1px solid #2f2f2f;
 | 
			
		||||
      border-radius: 4px;
 | 
			
		||||
      overflow: hidden;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    &__minimap-mask {
 | 
			
		||||
      fill: rgba(28, 28, 28, 0.75);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    &__controls {
 | 
			
		||||
      filter: brightness(1.5);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    &__minimap-node {
 | 
			
		||||
      fill: #ffb03a;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  .context-menu-active {
 | 
			
		||||
    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,24 +1,32 @@
 | 
			
		||||
 | 
			
		||||
@import './eve-common-variables';
 | 
			
		||||
@import './eve-common';
 | 
			
		||||
@import url('https://fonts.googleapis.com/css2?family=Oxygen:wght@300;400;700&display=swap');
 | 
			
		||||
 | 
			
		||||
$homeBase: rgb(197, 253, 67);
 | 
			
		||||
$homeAlpha: rgba(197, 253, 67, 0.32);
 | 
			
		||||
$homeDark30: darken($homeBase, 30%);
 | 
			
		||||
 | 
			
		||||
.pathfinder-theme {
 | 
			
		||||
  /* -- Override values from the default theme -- */
 | 
			
		||||
  --rf-bg-color: #000000;
 | 
			
		||||
  --rf-soft-bg-color: #282828;
 | 
			
		||||
 | 
			
		||||
  --rf-node-bg-color: #202020;
 | 
			
		||||
  --rf-node-soft-bg-color: #313335;
 | 
			
		||||
  --rf-node-font-weight: bold;
 | 
			
		||||
 | 
			
		||||
  --rf-text-color: #adadad;
 | 
			
		||||
  --tooltip-bg: #202020;
 | 
			
		||||
 | 
			
		||||
  --rf-region-name: var(--rf-text-color);
 | 
			
		||||
  --rf-custom-name: var(--rf-text-color);
 | 
			
		||||
  --rf-bg-variant: "lines";
 | 
			
		||||
  --rf-bg-gap: 32;
 | 
			
		||||
  --rf-bg-size: 1;
 | 
			
		||||
  --rf-bg-gap: 34;
 | 
			
		||||
  --rf-snap-size: 17;
 | 
			
		||||
  --rf-bg-pattern-color: #313131;
 | 
			
		||||
  --rf-local-counter-font-weight: 700;
 | 
			
		||||
 | 
			
		||||
  /* Additional node-specific overrides */
 | 
			
		||||
  --rf-node-line-height: normal;
 | 
			
		||||
  --rf-node-font-family: 'Oxygen', sans-serif;
 | 
			
		||||
  --rf-tag-color: #fbbf24;
 | 
			
		||||
 | 
			
		||||
  /* -- theme-specific variables -- */
 | 
			
		||||
  --eve-effect-pulsar: #428bca;
 | 
			
		||||
  --eve-effect-magnetar: #e06fdf;
 | 
			
		||||
  --eve-effect-wolfRayet: #e28a0d;
 | 
			
		||||
@@ -38,12 +46,9 @@
 | 
			
		||||
  --eve-wh-type-color-c13: #7986cb;
 | 
			
		||||
  --eve-wh-type-color-drifter: #44aa82;
 | 
			
		||||
 | 
			
		||||
  --rf-node-local-counter: #5cb85c;
 | 
			
		||||
  --rf-has-user-characters: #ffc75d;
 | 
			
		||||
 | 
			
		||||
  --rf-node-font-weight: bold;
 | 
			
		||||
  --rf-node-line-height: normal;
 | 
			
		||||
  --rf-node-font-family: 'Oxygen', sans-serif;
 | 
			
		||||
  --rf-node-text-color: var(--pf-text-color);
 | 
			
		||||
 | 
			
		||||
  --rf-tag-color: #fbbf24;
 | 
			
		||||
  --rf-has-user-characters: #5cb85c;
 | 
			
		||||
}
 | 
			
		||||
  --eve-solar-system-status-home:                #{$homeAlpha};
 | 
			
		||||
  --eve-solar-system-status-color-home:          #{$homeBase};
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,78 +1,30 @@
 | 
			
		||||
import 'react-grid-layout/css/styles.css';
 | 
			
		||||
import 'react-resizable/css/styles.css';
 | 
			
		||||
import { WidgetGridItem, WidgetsGrid } from '@/hooks/Mapper/components/mapInterface/components';
 | 
			
		||||
import {
 | 
			
		||||
  LocalCharacters,
 | 
			
		||||
  RoutesWidget,
 | 
			
		||||
  SystemInfo,
 | 
			
		||||
  SystemSignatures,
 | 
			
		||||
} from '@/hooks/Mapper/components/mapInterface/widgets';
 | 
			
		||||
import { useState } from 'react';
 | 
			
		||||
import { SESSION_KEY } from '@/hooks/Mapper/constants.ts';
 | 
			
		||||
// import { debounce } from 'lodash/debounce';
 | 
			
		||||
 | 
			
		||||
const DEFAULT_WINDOWS = [
 | 
			
		||||
  {
 | 
			
		||||
    name: 'info',
 | 
			
		||||
    rightOffset: 5,
 | 
			
		||||
    width: 5,
 | 
			
		||||
    height: 4,
 | 
			
		||||
    item: () => <SystemInfo />,
 | 
			
		||||
  },
 | 
			
		||||
  {
 | 
			
		||||
    name: 'local',
 | 
			
		||||
    rightOffset: 5,
 | 
			
		||||
    topOffset: 4,
 | 
			
		||||
    width: 5,
 | 
			
		||||
    height: 4,
 | 
			
		||||
    item: () => <LocalCharacters />,
 | 
			
		||||
  },
 | 
			
		||||
  { name: 'signatures', width: 8, height: 4, topOffset: 8, rightOffset: 12, item: () => <SystemSignatures /> },
 | 
			
		||||
  {
 | 
			
		||||
    name: 'routes',
 | 
			
		||||
    rightOffset: 0,
 | 
			
		||||
    topOffset: 8,
 | 
			
		||||
    width: 5,
 | 
			
		||||
    height: 6,
 | 
			
		||||
    item: () => <RoutesWidget />,
 | 
			
		||||
  },
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
const saveWindowsToLS = (toSaveItems: WidgetGridItem[]) => {
 | 
			
		||||
  // eslint-disable-next-line @typescript-eslint/no-unused-vars
 | 
			
		||||
  const out = toSaveItems.map(({ item, ...rest }) => rest);
 | 
			
		||||
  localStorage.setItem(SESSION_KEY.windows, JSON.stringify(out));
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const restoreWindowsFromLS = (): WidgetGridItem[] => {
 | 
			
		||||
  // eslint-disable-next-line @typescript-eslint/no-unused-vars
 | 
			
		||||
  const raw = localStorage.getItem(SESSION_KEY.windows);
 | 
			
		||||
  if (!raw) {
 | 
			
		||||
    console.warn('No windows found in local storage!!');
 | 
			
		||||
    return DEFAULT_WINDOWS;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // eslint-disable-next-line no-debugger
 | 
			
		||||
  const out = (JSON.parse(raw) as Omit<WidgetGridItem, 'item'>[])
 | 
			
		||||
    .filter(x => DEFAULT_WINDOWS.find(def => def.name === x.name))
 | 
			
		||||
    .map(x => {
 | 
			
		||||
      const windowItem = DEFAULT_WINDOWS.find(def => def.name === x.name)?.item;
 | 
			
		||||
      return { ...x, item: windowItem! };
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
  return out;
 | 
			
		||||
};
 | 
			
		||||
import { useMemo } from 'react';
 | 
			
		||||
import { WindowManager } from '@/hooks/Mapper/components/ui-kit/WindowManager';
 | 
			
		||||
import { DEFAULT_WIDGETS } from '@/hooks/Mapper/components/mapInterface/constants.tsx';
 | 
			
		||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
 | 
			
		||||
export const MapInterface = () => {
 | 
			
		||||
  const [items, setItems] = useState<WidgetGridItem[]>(restoreWindowsFromLS);
 | 
			
		||||
  // const [items, setItems] = useState<WindowProps[]>(restoreWindowsFromLS);
 | 
			
		||||
  const { windowsSettings, updateWidgetSettings } = useMapRootState();
 | 
			
		||||
 | 
			
		||||
  const items = useMemo(() => {
 | 
			
		||||
    return windowsSettings.windows
 | 
			
		||||
      .map(x => {
 | 
			
		||||
        const content = DEFAULT_WIDGETS.find(y => y.id === x.id)?.content;
 | 
			
		||||
        return {
 | 
			
		||||
          ...x,
 | 
			
		||||
          content: content!,
 | 
			
		||||
        };
 | 
			
		||||
      })
 | 
			
		||||
      .filter(x => windowsSettings.visible.some(j => x.id === j));
 | 
			
		||||
  }, [windowsSettings]);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <WidgetsGrid
 | 
			
		||||
      items={items}
 | 
			
		||||
      onChange={x => {
 | 
			
		||||
        saveWindowsToLS(x);
 | 
			
		||||
        setItems(x);
 | 
			
		||||
      }}
 | 
			
		||||
    <WindowManager
 | 
			
		||||
      windows={items}
 | 
			
		||||
      viewPort={windowsSettings.viewPort}
 | 
			
		||||
      dragSelector=".react-grid-dragHandleExample"
 | 
			
		||||
      onChange={updateWidgetSettings}
 | 
			
		||||
    />
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
 
 | 
			
		||||
@@ -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,95 @@
 | 
			
		||||
import classes from './MarkdownComment.module.scss';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
import Markdown from 'react-markdown';
 | 
			
		||||
import remarkGfm from 'remark-gfm';
 | 
			
		||||
import { InfoDrawer, TimeAgo, TooltipPosition, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
 | 
			
		||||
import remarkBreaks from 'remark-breaks';
 | 
			
		||||
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';
 | 
			
		||||
 | 
			
		||||
const TOOLTIP_PROPS = { content: 'Remove comment', position: TooltipPosition.top };
 | 
			
		||||
const REMARK_PLUGINS = [remarkGfm, remarkBreaks];
 | 
			
		||||
 | 
			
		||||
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 cpRemoveBtnRef = useRef<HTMLElement>();
 | 
			
		||||
  const [cpRemoveVisible, setCpRemoveVisible] = useState(false);
 | 
			
		||||
 | 
			
		||||
  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), []);
 | 
			
		||||
 | 
			
		||||
  const handleShowCP = useCallback(() => setCpRemoveVisible(true), []);
 | 
			
		||||
  const handleHideCP = useCallback(() => setCpRemoveVisible(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={cpRemoveBtnRef}>
 | 
			
		||||
                    <WdImgButton
 | 
			
		||||
                      className={clsx(PrimeIcons.TRASH, 'hover:text-red-400')}
 | 
			
		||||
                      tooltip={TOOLTIP_PROPS}
 | 
			
		||||
                      onClick={handleShowCP}
 | 
			
		||||
                    />
 | 
			
		||||
                  </div>
 | 
			
		||||
                )}
 | 
			
		||||
              </div>
 | 
			
		||||
            </WdTransition>
 | 
			
		||||
          </div>
 | 
			
		||||
        }
 | 
			
		||||
      >
 | 
			
		||||
        <Markdown remarkPlugins={REMARK_PLUGINS}>{text}</Markdown>
 | 
			
		||||
      </InfoDrawer>
 | 
			
		||||
 | 
			
		||||
      <ConfirmPopup
 | 
			
		||||
        target={cpRemoveBtnRef.current}
 | 
			
		||||
        visible={cpRemoveVisible}
 | 
			
		||||
        onHide={handleHideCP}
 | 
			
		||||
        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,72 @@
 | 
			
		||||
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 {}
 | 
			
		||||
 | 
			
		||||
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>
 | 
			
		||||
      }
 | 
			
		||||
    />
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
@@ -0,0 +1 @@
 | 
			
		||||
export * from './CommentsEditor';
 | 
			
		||||
@@ -0,0 +1,40 @@
 | 
			
		||||
.CERoot {
 | 
			
		||||
  @apply border border-stone-400/30 rounded-[2px];
 | 
			
		||||
 | 
			
		||||
  :global {
 | 
			
		||||
    .cm-content {
 | 
			
		||||
      @apply bg-stone-600/40;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .cm-scroller {
 | 
			
		||||
      scrollbar-width: thin;
 | 
			
		||||
      scrollbar-color: rgba(255, 255, 255, 0.5) transparent;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .cm-scroller::-webkit-scrollbar {
 | 
			
		||||
      width: 10px;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .cm-scroller::-webkit-scrollbar-track {
 | 
			
		||||
      background: transparent;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .cm-scroller::-webkit-scrollbar-thumb {
 | 
			
		||||
      background-color: rgba(255, 255, 255, 0.5);
 | 
			
		||||
      border-radius: 5px;
 | 
			
		||||
      border: 2px solid transparent;
 | 
			
		||||
      background-clip: content-box;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .cm-scroller::-webkit-scrollbar-thumb:hover {
 | 
			
		||||
      background-color: rgba(255, 255, 255, 0.7);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .cm-scroller::-webkit-scrollbar-button {
 | 
			
		||||
      display: none;
 | 
			
		||||
      height: 0;
 | 
			
		||||
      width: 0;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,86 @@
 | 
			
		||||
import { ReactNode, useCallback, useRef, useState } from 'react';
 | 
			
		||||
import CodeMirror, { ViewPlugin } from '@uiw/react-codemirror';
 | 
			
		||||
import { markdown } from '@codemirror/lang-markdown';
 | 
			
		||||
import { oneDark } from '@codemirror/theme-one-dark';
 | 
			
		||||
import { EditorView, type ViewUpdate } from '@codemirror/view';
 | 
			
		||||
 | 
			
		||||
import classes from './MarkdownEditor.module.scss';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
 | 
			
		||||
// TODO special plugin which force CodeMirror using capture for paste event
 | 
			
		||||
const stopEventPropagationPlugin = ViewPlugin.fromClass(
 | 
			
		||||
  class {
 | 
			
		||||
    constructor(view: EditorView) {
 | 
			
		||||
      // @ts-ignore
 | 
			
		||||
      this.view = view;
 | 
			
		||||
 | 
			
		||||
      // @ts-ignore
 | 
			
		||||
      this.pasteHandler = (event: Event) => {
 | 
			
		||||
        event.stopPropagation();
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      // @ts-ignore
 | 
			
		||||
      view.dom.addEventListener('paste', this.pasteHandler);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    destroy() {
 | 
			
		||||
      // @ts-ignore
 | 
			
		||||
      this.view.dom.removeEventListener('paste', this.pasteHandler);
 | 
			
		||||
    }
 | 
			
		||||
  },
 | 
			
		||||
);
 | 
			
		||||
 | 
			
		||||
const CODE_MIRROR_EXTENSIONS = [
 | 
			
		||||
  markdown(),
 | 
			
		||||
  EditorView.lineWrapping,
 | 
			
		||||
  EditorView.theme({
 | 
			
		||||
    '&': { backgroundColor: 'transparent !important' },
 | 
			
		||||
    '& .cm-gutterElement': { display: 'none' },
 | 
			
		||||
  }),
 | 
			
		||||
  stopEventPropagationPlugin,
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
export interface MarkdownEditorProps {
 | 
			
		||||
  overlayContent?: ReactNode;
 | 
			
		||||
  value: string;
 | 
			
		||||
  onChange: (value: string) => void;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const MarkdownEditor = ({ value, onChange, overlayContent }: MarkdownEditorProps) => {
 | 
			
		||||
  const [hasShift, setHasShift] = useState(false);
 | 
			
		||||
 | 
			
		||||
  const refData = useRef({ onChange });
 | 
			
		||||
  refData.current = { onChange };
 | 
			
		||||
 | 
			
		||||
  const handleOnChange = useCallback((value: string, viewUpdate: ViewUpdate) => {
 | 
			
		||||
    // Rerender happens after change
 | 
			
		||||
    setTimeout(() => {
 | 
			
		||||
      const scrollDOM = viewUpdate.view.scrollDOM;
 | 
			
		||||
      setHasShift(scrollDOM.scrollHeight > scrollDOM.clientHeight);
 | 
			
		||||
    }, 0);
 | 
			
		||||
 | 
			
		||||
    refData.current.onChange(value);
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div className={clsx(classes.MarkdownEditor, 'relative')}>
 | 
			
		||||
      <CodeMirror
 | 
			
		||||
        value={value}
 | 
			
		||||
        height="70px"
 | 
			
		||||
        extensions={CODE_MIRROR_EXTENSIONS}
 | 
			
		||||
        className={classes.CERoot}
 | 
			
		||||
        theme={oneDark}
 | 
			
		||||
        onChange={handleOnChange}
 | 
			
		||||
        placeholder="Start typing..."
 | 
			
		||||
      />
 | 
			
		||||
      <div
 | 
			
		||||
        className={clsx('absolute top-0 left-0 h-full pointer-events-none', {
 | 
			
		||||
          'w-full': !hasShift,
 | 
			
		||||
          'w-[calc(100%-10px)]': hasShift,
 | 
			
		||||
        })}
 | 
			
		||||
      >
 | 
			
		||||
        {overlayContent}
 | 
			
		||||
      </div>
 | 
			
		||||
    </div>
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
@@ -0,0 +1 @@
 | 
			
		||||
export * from './MarkdownEditor.tsx';
 | 
			
		||||
@@ -1,64 +1,178 @@
 | 
			
		||||
import { useCallback, useRef } from 'react';
 | 
			
		||||
import { Dialog } from 'primereact/dialog';
 | 
			
		||||
import { useCallback, useEffect, useMemo, useRef } from 'react';
 | 
			
		||||
 | 
			
		||||
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
 | 
			
		||||
import { SystemSignature } from '@/hooks/Mapper/types';
 | 
			
		||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
import { CommandLinkSignatureToSystem } from '@/hooks/Mapper/types';
 | 
			
		||||
import { SystemSignaturesContent } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SystemSignaturesContent';
 | 
			
		||||
import { SHOW_DESCRIPTION_COLUMN_SETTING } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SystemSignatures';
 | 
			
		||||
import { useSystemInfo } from '@/hooks/Mapper/components/hooks';
 | 
			
		||||
import {
 | 
			
		||||
  Setting,
 | 
			
		||||
  COSMIC_SIGNATURE,
 | 
			
		||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SystemSignatureSettingsDialog';
 | 
			
		||||
import { SignatureGroup } from '@/hooks/Mapper/types';
 | 
			
		||||
  SOLAR_SYSTEM_CLASS_IDS,
 | 
			
		||||
  SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS,
 | 
			
		||||
  WORMHOLES_ADDITIONAL_INFO_BY_SHORT_NAME,
 | 
			
		||||
} from '@/hooks/Mapper/components/map/constants.ts';
 | 
			
		||||
import {
 | 
			
		||||
  SETTINGS_KEYS,
 | 
			
		||||
  SignatureSettingsType,
 | 
			
		||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
 | 
			
		||||
import { SystemSignaturesContent } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SystemSignaturesContent';
 | 
			
		||||
import { K162_TYPES_MAP } from '@/hooks/Mapper/constants.ts';
 | 
			
		||||
import { getWhSize } from '@/hooks/Mapper/helpers/getWhSize';
 | 
			
		||||
import { parseSignatureCustomInfo } from '@/hooks/Mapper/helpers/parseSignatureCustomInfo';
 | 
			
		||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
 | 
			
		||||
import { CommandLinkSignatureToSystem, SignatureGroup, SystemSignature, TimeStatus } from '@/hooks/Mapper/types';
 | 
			
		||||
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
 | 
			
		||||
 | 
			
		||||
const K162_SIGNATURE_TYPE = WORMHOLES_ADDITIONAL_INFO_BY_SHORT_NAME['K162'].shortName;
 | 
			
		||||
 | 
			
		||||
interface SystemLinkSignatureDialogProps {
 | 
			
		||||
  data: CommandLinkSignatureToSystem;
 | 
			
		||||
  setVisible: (visible: boolean) => void;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const signatureSettings: Setting[] = [
 | 
			
		||||
  { key: COSMIC_SIGNATURE, name: 'Show Cosmic Signatures', value: true },
 | 
			
		||||
  { key: SignatureGroup.Wormhole, name: 'Wormhole', value: true },
 | 
			
		||||
  { key: SHOW_DESCRIPTION_COLUMN_SETTING, name: 'Show Description Column', value: true, isFilter: false },
 | 
			
		||||
];
 | 
			
		||||
export const LINK_SIGNTATURE_SETTINGS: SignatureSettingsType = {
 | 
			
		||||
  [SETTINGS_KEYS.COSMIC_SIGNATURE]: true,
 | 
			
		||||
  [SETTINGS_KEYS.WORMHOLE]: true,
 | 
			
		||||
  [SETTINGS_KEYS.SHOW_DESCRIPTION_COLUMN]: true,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Extend the SignatureCustomInfo type to include k162Type
 | 
			
		||||
interface ExtendedSignatureCustomInfo {
 | 
			
		||||
  k162Type?: string;
 | 
			
		||||
  isEOL?: boolean;
 | 
			
		||||
  [key: string]: unknown;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const SystemLinkSignatureDialog = ({ data, setVisible }: SystemLinkSignatureDialogProps) => {
 | 
			
		||||
  const { outCommand } = useMapRootState();
 | 
			
		||||
  const {
 | 
			
		||||
    outCommand,
 | 
			
		||||
    data: { wormholes },
 | 
			
		||||
  } = useMapRootState();
 | 
			
		||||
 | 
			
		||||
  const ref = useRef({ outCommand });
 | 
			
		||||
  ref.current = { outCommand };
 | 
			
		||||
 | 
			
		||||
  // Get system info for the target system
 | 
			
		||||
  const { staticInfo: targetSystemInfo, dynamicInfo: targetSystemDynamicInfo } = useSystemInfo({
 | 
			
		||||
    systemId: `${data.solar_system_target}`,
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  // Get the system class group for the target system
 | 
			
		||||
  const targetSystemClassGroup = useMemo(() => {
 | 
			
		||||
    if (!targetSystemInfo) return null;
 | 
			
		||||
    const systemClassId = targetSystemInfo.system_class;
 | 
			
		||||
 | 
			
		||||
    const systemClassKey = Object.keys(SOLAR_SYSTEM_CLASS_IDS).find(
 | 
			
		||||
      key => SOLAR_SYSTEM_CLASS_IDS[key as keyof typeof SOLAR_SYSTEM_CLASS_IDS] === systemClassId,
 | 
			
		||||
    );
 | 
			
		||||
 | 
			
		||||
    if (!systemClassKey) return null;
 | 
			
		||||
 | 
			
		||||
    return (
 | 
			
		||||
      SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS[systemClassKey as keyof typeof SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS] || null
 | 
			
		||||
    );
 | 
			
		||||
  }, [targetSystemInfo]);
 | 
			
		||||
 | 
			
		||||
  const handleHide = useCallback(() => {
 | 
			
		||||
    setVisible(false);
 | 
			
		||||
  }, [setVisible]);
 | 
			
		||||
 | 
			
		||||
  const handleSelect = useCallback(
 | 
			
		||||
  const filterSignature = useCallback(
 | 
			
		||||
    (signature: SystemSignature) => {
 | 
			
		||||
      if (signature.group !== SignatureGroup.Wormhole || !targetSystemClassGroup) {
 | 
			
		||||
        return true;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (!signature.type) {
 | 
			
		||||
        return true;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (signature.type === K162_SIGNATURE_TYPE) {
 | 
			
		||||
        // Parse the custom info to see if the user has specified what class this K162 leads to
 | 
			
		||||
        const customInfo = parseSignatureCustomInfo(signature.custom_info) as ExtendedSignatureCustomInfo;
 | 
			
		||||
 | 
			
		||||
        // If the user has specified a k162Type for this K162
 | 
			
		||||
        if (customInfo.k162Type) {
 | 
			
		||||
          // Get the K162 type information
 | 
			
		||||
          const k162TypeInfo = K162_TYPES_MAP[customInfo.k162Type];
 | 
			
		||||
 | 
			
		||||
          if (k162TypeInfo) {
 | 
			
		||||
            // Check if the k162Type matches our target system class
 | 
			
		||||
            return customInfo.k162Type === targetSystemClassGroup;
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // If no k162Type is specified or we couldn't find type info, allow it
 | 
			
		||||
        return true;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Find the wormhole data for this signature type
 | 
			
		||||
      const wormholeData = wormholes.find(wh => wh.name === signature.type);
 | 
			
		||||
      if (!wormholeData) {
 | 
			
		||||
        return true; // If we don't know the destination, don't filter it out
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      // Get the destination system class from the wormhole data
 | 
			
		||||
      const destinationClass = wormholeData.dest;
 | 
			
		||||
 | 
			
		||||
      // Check if the destination class matches the target system class
 | 
			
		||||
      const isMatch = destinationClass === targetSystemClassGroup;
 | 
			
		||||
      return isMatch;
 | 
			
		||||
    },
 | 
			
		||||
    [targetSystemClassGroup, wormholes],
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  const handleSelect = useCallback(
 | 
			
		||||
    async (signature: SystemSignature) => {
 | 
			
		||||
      if (!signature) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const { outCommand } = ref.current;
 | 
			
		||||
 | 
			
		||||
      outCommand({
 | 
			
		||||
      await outCommand({
 | 
			
		||||
        type: OutCommand.linkSignatureToSystem,
 | 
			
		||||
        data: {
 | 
			
		||||
          ...data,
 | 
			
		||||
          signature_eve_id: signature.eve_id,
 | 
			
		||||
        },
 | 
			
		||||
      });
 | 
			
		||||
 | 
			
		||||
      if (parseSignatureCustomInfo(signature.custom_info).isEOL === true) {
 | 
			
		||||
        await outCommand({
 | 
			
		||||
          type: OutCommand.updateConnectionTimeStatus,
 | 
			
		||||
          data: {
 | 
			
		||||
            source: data.solar_system_source,
 | 
			
		||||
            target: data.solar_system_target,
 | 
			
		||||
            value: TimeStatus.eol,
 | 
			
		||||
          },
 | 
			
		||||
        });
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const whShipSize = getWhSize(wormholes, signature.type);
 | 
			
		||||
      if (whShipSize) {
 | 
			
		||||
        await outCommand({
 | 
			
		||||
          type: OutCommand.updateConnectionShipSizeType,
 | 
			
		||||
          data: {
 | 
			
		||||
            source: data.solar_system_source,
 | 
			
		||||
            target: data.solar_system_target,
 | 
			
		||||
            value: whShipSize,
 | 
			
		||||
          },
 | 
			
		||||
        });
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      setVisible(false);
 | 
			
		||||
    },
 | 
			
		||||
    [data, setVisible],
 | 
			
		||||
    [data, setVisible, wormholes],
 | 
			
		||||
  );
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    if (!targetSystemDynamicInfo) {
 | 
			
		||||
      handleHide();
 | 
			
		||||
    }
 | 
			
		||||
  }, [targetSystemDynamicInfo]);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <Dialog
 | 
			
		||||
      header="Select signature to link"
 | 
			
		||||
      visible
 | 
			
		||||
      draggable={false}
 | 
			
		||||
      draggable={true}
 | 
			
		||||
      style={{ width: '500px' }}
 | 
			
		||||
      onHide={handleHide}
 | 
			
		||||
      contentClassName="!p-0"
 | 
			
		||||
@@ -66,9 +180,10 @@ export const SystemLinkSignatureDialog = ({ data, setVisible }: SystemLinkSignat
 | 
			
		||||
      <SystemSignaturesContent
 | 
			
		||||
        systemId={`${data.solar_system_source}`}
 | 
			
		||||
        hideLinkedSignatures
 | 
			
		||||
        settings={signatureSettings}
 | 
			
		||||
        settings={LINK_SIGNTATURE_SETTINGS}
 | 
			
		||||
        onSelect={handleSelect}
 | 
			
		||||
        selectable={true}
 | 
			
		||||
        filterSignature={filterSignature}
 | 
			
		||||
      />
 | 
			
		||||
    </Dialog>
 | 
			
		||||
  );
 | 
			
		||||
 
 | 
			
		||||
@@ -10,6 +10,7 @@ import { OutCommand } from '@/hooks/Mapper/types';
 | 
			
		||||
import { IconField } from 'primereact/iconfield';
 | 
			
		||||
import { TooltipPosition, WdImageSize, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
 | 
			
		||||
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager.ts';
 | 
			
		||||
import { getSystemStaticInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
 | 
			
		||||
 | 
			
		||||
interface SystemSettingsDialog {
 | 
			
		||||
  systemId: string;
 | 
			
		||||
@@ -26,6 +27,7 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
 | 
			
		||||
  const isTempSystemNameEnabled = useMapGetOption('show_temp_system_name') === 'true';
 | 
			
		||||
 | 
			
		||||
  const system = getSystemById(systems, systemId);
 | 
			
		||||
  const systemStaticInfo = getSystemStaticInfo(systemId);
 | 
			
		||||
 | 
			
		||||
  const [name, setName] = useState('');
 | 
			
		||||
  const [label, setLabel] = useState('');
 | 
			
		||||
@@ -33,11 +35,11 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
 | 
			
		||||
  const [description, setDescription] = useState('');
 | 
			
		||||
  const inputRef = useRef<HTMLInputElement>();
 | 
			
		||||
 | 
			
		||||
  const ref = useRef({ name, description, temporaryName, label, outCommand, systemId, system });
 | 
			
		||||
  ref.current = { name, description, label, temporaryName, outCommand, systemId, system };
 | 
			
		||||
  const ref = useRef({ name, description, temporaryName, label, outCommand, systemId, system, systemStaticInfo });
 | 
			
		||||
  ref.current = { name, description, label, temporaryName, outCommand, systemId, system, systemStaticInfo };
 | 
			
		||||
 | 
			
		||||
  const handleSave = useCallback(() => {
 | 
			
		||||
    const { name, description, label, temporaryName, outCommand, systemId, system } = ref.current;
 | 
			
		||||
    const { name, description, label, temporaryName, outCommand, systemId, system, systemStaticInfo } = ref.current;
 | 
			
		||||
 | 
			
		||||
    const outLabel = new LabelsManager(system?.labels ?? '');
 | 
			
		||||
    outLabel.updateCustomLabel(label);
 | 
			
		||||
@@ -62,7 +64,7 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
 | 
			
		||||
      type: OutCommand.updateSystemName,
 | 
			
		||||
      data: {
 | 
			
		||||
        system_id: systemId,
 | 
			
		||||
        value: name.trim() || system?.system_static_info.solar_system_name,
 | 
			
		||||
        value: name.trim() || systemStaticInfo?.solar_system_name,
 | 
			
		||||
      },
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
@@ -78,11 +80,11 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
 | 
			
		||||
  }, [setVisible]);
 | 
			
		||||
 | 
			
		||||
  const handleResetSystemName = useCallback(() => {
 | 
			
		||||
    const { system } = ref.current;
 | 
			
		||||
    if (!system) {
 | 
			
		||||
    const { systemStaticInfo } = ref.current;
 | 
			
		||||
    if (!systemStaticInfo) {
 | 
			
		||||
      return;
 | 
			
		||||
    }
 | 
			
		||||
    setName(system.system_static_info.solar_system_name);
 | 
			
		||||
    setName(systemStaticInfo.solar_system_name);
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
  const onShow = useCallback(() => {
 | 
			
		||||
@@ -130,7 +132,7 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
 | 
			
		||||
              <label htmlFor="username">Custom name</label>
 | 
			
		||||
 | 
			
		||||
              <IconField>
 | 
			
		||||
                {name !== system?.system_static_info.solar_system_name && (
 | 
			
		||||
                {name !== systemStaticInfo?.solar_system_name && (
 | 
			
		||||
                  <WdImgButton
 | 
			
		||||
                    className="pi pi-undo"
 | 
			
		||||
                    textSize={WdImageSize.large}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
.root {
 | 
			
		||||
  padding-bottom: 5px;
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.Header {
 | 
			
		||||
 
 | 
			
		||||
@@ -2,15 +2,18 @@ import React from 'react';
 | 
			
		||||
 | 
			
		||||
import classes from './Widget.module.scss';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
import { WithChildren } from '@/hooks/Mapper/types/common.ts';
 | 
			
		||||
 | 
			
		||||
export interface WidgetProps {
 | 
			
		||||
export type WidgetProps = {
 | 
			
		||||
  label: React.ReactNode | string;
 | 
			
		||||
  children?: React.ReactNode;
 | 
			
		||||
}
 | 
			
		||||
  windowId?: string;
 | 
			
		||||
  contentClassName?: string;
 | 
			
		||||
} & WithChildren;
 | 
			
		||||
 | 
			
		||||
export const Widget = ({ label, children }: WidgetProps) => {
 | 
			
		||||
export const Widget = ({ label, children, windowId, contentClassName }: WidgetProps) => {
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
      data-window-id={windowId}
 | 
			
		||||
      className={clsx(
 | 
			
		||||
        classes.root,
 | 
			
		||||
        'flex flex-col w-full h-full rounded',
 | 
			
		||||
@@ -32,7 +35,7 @@ export const Widget = ({ label, children }: WidgetProps) => {
 | 
			
		||||
        {label}
 | 
			
		||||
      </div>
 | 
			
		||||
      <div
 | 
			
		||||
        className={clsx(classes.Content, 'overflow-auto', 'bg-opacity-5  custom-scrollbar')}
 | 
			
		||||
        className={clsx(classes.Content, 'overflow-auto', 'bg-opacity-5 custom-scrollbar', contentClassName)}
 | 
			
		||||
        style={{ flexGrow: 1 }}
 | 
			
		||||
        onContextMenu={e => {
 | 
			
		||||
          e.preventDefault();
 | 
			
		||||
 
 | 
			
		||||
@@ -1,37 +0,0 @@
 | 
			
		||||
.GridLayoutWrapper {
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  height: 100% !important;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.GridLayout {
 | 
			
		||||
  width: 100%;
 | 
			
		||||
  height: 100% !important;
 | 
			
		||||
  pointer-events: none;
 | 
			
		||||
 | 
			
		||||
  & > div {
 | 
			
		||||
    pointer-events: initial;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  :global {
 | 
			
		||||
    .react-resizable-handle::after {
 | 
			
		||||
      border-color: #696969 !important;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .react-grid-placeholder {
 | 
			
		||||
      background-color: rgba(147, 147, 147, 0.3);
 | 
			
		||||
      //filter: blur(5px);
 | 
			
		||||
      border: 2px dashed #b6b6b6;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .react-grid-item {
 | 
			
		||||
      transition-property: none !important;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    .react-grid-item.cssTransforms {
 | 
			
		||||
      transition-property: none !important;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@@ -1,196 +0,0 @@
 | 
			
		||||
import React, { useEffect, useRef, useState } from 'react';
 | 
			
		||||
 | 
			
		||||
import classes from './WidgetsGrid.module.scss';
 | 
			
		||||
import { ItemCallback, Layouts, Responsive, WidthProvider } from 'react-grid-layout';
 | 
			
		||||
import clsx from 'clsx';
 | 
			
		||||
import usePageVisibility from '@/hooks/Mapper/hooks/usePageVisibility.ts';
 | 
			
		||||
 | 
			
		||||
const ResponsiveGridLayout = WidthProvider(Responsive);
 | 
			
		||||
 | 
			
		||||
const colSize = 50;
 | 
			
		||||
const initState = { breakpoints: 100, cols: 2 };
 | 
			
		||||
 | 
			
		||||
export type WidgetGridItem = {
 | 
			
		||||
  rightOffset?: number;
 | 
			
		||||
  leftOffset?: number;
 | 
			
		||||
  topOffset?: number;
 | 
			
		||||
  width: number;
 | 
			
		||||
  height: number;
 | 
			
		||||
  name: string;
 | 
			
		||||
  item: () => React.ReactNode;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export interface WidgetsGridProps {
 | 
			
		||||
  items: WidgetGridItem[];
 | 
			
		||||
  onChange: (items: WidgetGridItem[]) => void;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export const WidgetsGrid = ({ items, onChange }: WidgetsGridProps) => {
 | 
			
		||||
  const containerRef = useRef<HTMLDivElement>(null);
 | 
			
		||||
  const [, setKey] = useState(0);
 | 
			
		||||
  const [callRerenderOfGrid, setCallRerenderOfGrid] = useState(0);
 | 
			
		||||
 | 
			
		||||
  const isTabVisible = usePageVisibility();
 | 
			
		||||
 | 
			
		||||
  const refAll = useRef({
 | 
			
		||||
    isReady: false,
 | 
			
		||||
    layouts: {
 | 
			
		||||
      lg: [
 | 
			
		||||
        // { i: 'a', w: 4, h: 16, x: 22, y: 0 },
 | 
			
		||||
        // { i: 'b', w: 5, h: 10, x: 17, y: 0 },
 | 
			
		||||
      ],
 | 
			
		||||
    } as Layouts,
 | 
			
		||||
    breakpoints: { lg: 100, md: 0, sm: 0, xs: 0, xxs: 0 },
 | 
			
		||||
    cols: { lg: 26, md: 0, sm: 0, xs: 0, xxs: 0 },
 | 
			
		||||
    containerWidth: 0,
 | 
			
		||||
    colsPrev: 26,
 | 
			
		||||
    needPostProcess: false,
 | 
			
		||||
    items: [...items],
 | 
			
		||||
  });
 | 
			
		||||
 | 
			
		||||
  // TODO
 | 
			
		||||
  //  1. onLayoutChange (original) not calling when we change x of any widget
 | 
			
		||||
  //  2. setKey need no call rerender for update props
 | 
			
		||||
  const onLayoutChange: ItemCallback = (newItems, _, newItem) => {
 | 
			
		||||
    const updatedItems = newItems.map(item => {
 | 
			
		||||
      const toLeft = (item.x + item.w / 2) / refAll.current.cols.lg <= 0.5;
 | 
			
		||||
      const original = refAll.current.items.find(x => x.name === item.i)!;
 | 
			
		||||
 | 
			
		||||
      return {
 | 
			
		||||
        ...original,
 | 
			
		||||
        width: item.w,
 | 
			
		||||
        height: item.h,
 | 
			
		||||
        leftOffset: toLeft ? item.x : undefined,
 | 
			
		||||
        rightOffset: !toLeft ? refAll.current.cols.lg - (item.x + item.w) : undefined,
 | 
			
		||||
        topOffset: item.y,
 | 
			
		||||
      };
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    const sortedItems = [
 | 
			
		||||
      ...updatedItems.filter(x => x.name !== newItem.i),
 | 
			
		||||
      updatedItems.find(x => x.name === newItem.i)!,
 | 
			
		||||
    ];
 | 
			
		||||
 | 
			
		||||
    refAll.current.layouts = {
 | 
			
		||||
      lg: [...newItems.filter(x => x.i !== newItem.i), newItem],
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    onChange(sortedItems);
 | 
			
		||||
    setKey(x => x + 1);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    refAll.current.items = [...items];
 | 
			
		||||
    setKey(x => x + 1);
 | 
			
		||||
  }, [items]);
 | 
			
		||||
 | 
			
		||||
  // TODO
 | 
			
		||||
  //  1. Unknown why but if we set layout and cols both instantly it not help...
 | 
			
		||||
  //  1.2 it means that we should make report... until we will send new key on window resize
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    const updateItems = () => {
 | 
			
		||||
      if (!containerRef.current) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const { width } = containerRef.current.getBoundingClientRect();
 | 
			
		||||
      const newColsCount = (width - (width % colSize)) / colSize;
 | 
			
		||||
 | 
			
		||||
      refAll.current.layouts = {
 | 
			
		||||
        lg: refAll.current.items.map(({ name, width, height, rightOffset, leftOffset, topOffset = 0 }) => {
 | 
			
		||||
          return {
 | 
			
		||||
            i: name,
 | 
			
		||||
            x: rightOffset != null ? newColsCount - width - rightOffset : leftOffset ?? 0,
 | 
			
		||||
            y: topOffset,
 | 
			
		||||
            w: width,
 | 
			
		||||
            h: height,
 | 
			
		||||
          };
 | 
			
		||||
        }),
 | 
			
		||||
      };
 | 
			
		||||
      refAll.current.cols = { lg: newColsCount, md: 0, sm: 0, xs: 0, xxs: 0 };
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    const updateContainerWidth = () => {
 | 
			
		||||
      if (!containerRef.current) {
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      const { width } = containerRef.current.getBoundingClientRect();
 | 
			
		||||
 | 
			
		||||
      refAll.current.containerWidth = width;
 | 
			
		||||
      const newColsCount = (width - (width % colSize)) / colSize;
 | 
			
		||||
 | 
			
		||||
      if (width <= 100 || refAll.current.cols.lg === newColsCount) {
 | 
			
		||||
        return false;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      if (!refAll.current.isReady) {
 | 
			
		||||
        updateItems();
 | 
			
		||||
        setCallRerenderOfGrid(x => x + 1);
 | 
			
		||||
        refAll.current.isReady = true;
 | 
			
		||||
        return;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      refAll.current.layouts = {
 | 
			
		||||
        lg: refAll.current.layouts.lg.map(lgEl => {
 | 
			
		||||
          const toLeft = (lgEl.x + lgEl.w / 2) / refAll.current.cols.lg <= 0.5;
 | 
			
		||||
          const next = {
 | 
			
		||||
            ...lgEl,
 | 
			
		||||
            x: toLeft ? lgEl.x : newColsCount - (refAll.current.cols.lg - lgEl.x),
 | 
			
		||||
          };
 | 
			
		||||
          return next;
 | 
			
		||||
        }),
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      refAll.current.cols = { lg: newColsCount, md: 0, sm: 0, xs: 0, xxs: 0 };
 | 
			
		||||
      setCallRerenderOfGrid(x => x + 1);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    setTimeout(() => updateContainerWidth(), 100);
 | 
			
		||||
 | 
			
		||||
    const withRerender = () => {
 | 
			
		||||
      updateContainerWidth();
 | 
			
		||||
      setCallRerenderOfGrid(x => x + 1);
 | 
			
		||||
    };
 | 
			
		||||
 | 
			
		||||
    window.addEventListener('resize', withRerender);
 | 
			
		||||
    return () => {
 | 
			
		||||
      window.removeEventListener('resize', withRerender);
 | 
			
		||||
    };
 | 
			
		||||
  }, []);
 | 
			
		||||
 | 
			
		||||
  const isNotSet = initState.cols === refAll.current.cols.lg;
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div ref={containerRef} className={clsx(classes.GridLayoutWrapper, 'relative p-4')}>
 | 
			
		||||
      {!isNotSet && isTabVisible && (
 | 
			
		||||
        <ResponsiveGridLayout
 | 
			
		||||
          key={callRerenderOfGrid}
 | 
			
		||||
          className={classes.GridLayout}
 | 
			
		||||
          layouts={refAll.current.layouts}
 | 
			
		||||
          breakpoints={refAll.current.breakpoints}
 | 
			
		||||
          cols={refAll.current.cols}
 | 
			
		||||
          rowHeight={30}
 | 
			
		||||
          width={refAll.current.containerWidth}
 | 
			
		||||
          preventCollision={true}
 | 
			
		||||
          compactType={null}
 | 
			
		||||
          allowOverlap
 | 
			
		||||
          onDragStop={onLayoutChange}
 | 
			
		||||
          onResizeStop={onLayoutChange}
 | 
			
		||||
          // onResizeStart={onLayoutChange}
 | 
			
		||||
          // onDragStart={onLayoutChange}
 | 
			
		||||
          isBounded
 | 
			
		||||
          containerPadding={[0, 0]}
 | 
			
		||||
          resizeHandles={['sw', 'se']}
 | 
			
		||||
          draggableHandle=".react-grid-dragHandleExample"
 | 
			
		||||
        >
 | 
			
		||||
          {refAll.current.items.map(x => (
 | 
			
		||||
            <div key={x.name} className="grid-item">
 | 
			
		||||
              {x.item()}
 | 
			
		||||
            </div>
 | 
			
		||||
          ))}
 | 
			
		||||
        </ResponsiveGridLayout>
 | 
			
		||||
      )}
 | 
			
		||||
    </div>
 | 
			
		||||
  );
 | 
			
		||||
};
 | 
			
		||||
@@ -1 +0,0 @@
 | 
			
		||||
export * from './WidgetsGrid';
 | 
			
		||||
@@ -1,5 +1,4 @@
 | 
			
		||||
export * from './Widget';
 | 
			
		||||
export * from './WidgetsGrid';
 | 
			
		||||
export * from './SystemSettingsDialog';
 | 
			
		||||
export * from './SystemCustomLabelDialog';
 | 
			
		||||
export * from './SystemLinkSignatureDialog';
 | 
			
		||||
 
 | 
			
		||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user