mirror of
https://github.com/wanderer-industries/wanderer
synced 2025-11-14 21:26:10 +00:00
Compare commits
198 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f58c52d26b | ||
|
|
41e7739461 | ||
|
|
332152b677 | ||
|
|
85b49fe1f0 | ||
|
|
e7924532be | ||
|
|
475d950ad6 | ||
|
|
e6cfb29c6f | ||
|
|
dee6e86db1 | ||
|
|
72f088331f | ||
|
|
bbf536d10e | ||
|
|
149ac98297 | ||
|
|
b90a2910c9 | ||
|
|
c4da8a3a8d | ||
|
|
3ca75583d2 | ||
|
|
5f4607ae6f | ||
|
|
d880c6873f | ||
|
|
937649b2ed | ||
|
|
78e912c886 | ||
|
|
696c7d2cd1 | ||
|
|
49c0cb026b | ||
|
|
7091341b4b | ||
|
|
8795ce6af3 | ||
|
|
239b34d15a | ||
|
|
729a5ad1a9 | ||
|
|
8febc2476b | ||
|
|
84b1317927 | ||
|
|
bfb504e5db | ||
|
|
9975deacfb | ||
|
|
f77f071003 | ||
|
|
4a8d55e83d | ||
|
|
9a99f40e2a | ||
|
|
428fa8035c | ||
|
|
745f3dee17 | ||
|
|
9907cc1875 | ||
|
|
130cd780a2 | ||
|
|
a808e5d1a5 | ||
|
|
b926117e26 | ||
|
|
fdf238accf | ||
|
|
4e1c27e8a3 | ||
|
|
a3e51a0ac5 | ||
|
|
d27bb0d54f | ||
|
|
f6a750f06b | ||
|
|
c4e2f63e69 | ||
|
|
675ffc8f42 | ||
|
|
cdc4221175 | ||
|
|
843f3363fd | ||
|
|
17653a6374 | ||
|
|
7d860533a0 | ||
|
|
0c751b3ced | ||
|
|
80a522ab06 | ||
|
|
0718d76e00 | ||
|
|
a4887c5358 | ||
|
|
2ad5e122ff | ||
|
|
832d874a0e | ||
|
|
6a133d4dbd | ||
|
|
d96dfa63c9 | ||
|
|
d5c8c05426 | ||
|
|
b6bb4647c7 | ||
|
|
a81f06b743 | ||
|
|
cb41a33546 | ||
|
|
005068de9c | ||
|
|
d8c7b1e070 | ||
|
|
4835dfcc42 | ||
|
|
15bceb09a2 | ||
|
|
13e818abfd | ||
|
|
9c5f6049b5 | ||
|
|
2095b619a4 | ||
|
|
df155cbc1b | ||
|
|
3781729fd1 | ||
|
|
d03c634ec0 | ||
|
|
93c979c218 | ||
|
|
90fef94583 | ||
|
|
0b8eec2263 | ||
|
|
9511af4e6d | ||
|
|
7deaf1fd9f | ||
|
|
43cc5bd520 | ||
|
|
68b58aa520 | ||
|
|
dbadd09af3 | ||
|
|
fcbe2c754f | ||
|
|
ad4580677b | ||
|
|
01a6cc7d92 | ||
|
|
95ce95a187 | ||
|
|
ce8e6fbfb0 | ||
|
|
a20eaed76b | ||
|
|
419af72028 | ||
|
|
8e499522f6 | ||
|
|
84321b847e | ||
|
|
c969a4d465 | ||
|
|
0e12c850b6 | ||
|
|
442835dd9b | ||
|
|
b4ff99cb2e | ||
|
|
aa0ecbc998 | ||
|
|
cc412e93c0 | ||
|
|
1d36fadbfa | ||
|
|
56182bd87d | ||
|
|
d290ff92b3 | ||
|
|
298c5fd3b8 | ||
|
|
e365c43781 | ||
|
|
23a9f22ef4 | ||
|
|
242f437237 | ||
|
|
2eae8cffdb | ||
|
|
68ab3d4f72 | ||
|
|
1ea805aff0 | ||
|
|
6ce45349dc | ||
|
|
8f20cd9863 | ||
|
|
4ed0e85680 | ||
|
|
8ce9eb9955 | ||
|
|
363330f3d1 | ||
|
|
fbf9c5ddd6 | ||
|
|
fbf2ee314c | ||
|
|
c9f83fb419 | ||
|
|
9737d91e16 | ||
|
|
2f672ae970 | ||
|
|
25339546c6 | ||
|
|
912cad42ac | ||
|
|
b3752c8d8f | ||
|
|
e8a11333f2 | ||
|
|
8bb6d09e6e | ||
|
|
56bf955297 | ||
|
|
ef6c08dfe8 | ||
|
|
495c3e1cd7 | ||
|
|
9a5fe3d744 | ||
|
|
72607cae4d | ||
|
|
4891cdb04d | ||
|
|
d214881720 | ||
|
|
e66c125dbf | ||
|
|
9862bcfa05 | ||
|
|
0ac5451bef | ||
|
|
669479b815 | ||
|
|
2721130566 | ||
|
|
6e33ad943f | ||
|
|
f4b7357802 | ||
|
|
7a404a7e6a | ||
|
|
5158700a79 | ||
|
|
41d10c1b47 | ||
|
|
3aaac91f07 | ||
|
|
ea7ff080b8 | ||
|
|
b5270958eb | ||
|
|
b0a38eab8c | ||
|
|
0a478e82ba | ||
|
|
02d97a009c | ||
|
|
33940cdb9b | ||
|
|
7a63f9ee6b | ||
|
|
89b41fff59 | ||
|
|
7a15f71528 | ||
|
|
cdce2f8761 | ||
|
|
a2470bbe47 | ||
|
|
dbdf1ddce0 | ||
|
|
f767e42e6f | ||
|
|
3051eb6369 | ||
|
|
a41faddca3 | ||
|
|
469038730e | ||
|
|
b1fe5d2453 | ||
|
|
f43e717da0 | ||
|
|
95c8d4eef8 | ||
|
|
747ca0ee82 | ||
|
|
35a0184ec3 | ||
|
|
96e1e5328c | ||
|
|
7f98d6a0d8 | ||
|
|
0194e25696 | ||
|
|
189442e50f | ||
|
|
d9bed070ec | ||
|
|
a6193da8b5 | ||
|
|
50d35b207d | ||
|
|
19eb45bfa1 | ||
|
|
01e0b24d9d | ||
|
|
3c8024b16c | ||
|
|
4c0ad0dd66 | ||
|
|
501840086b | ||
|
|
240b180857 | ||
|
|
2bc5d0aaea | ||
|
|
df66aa79b8 | ||
|
|
6ea6a59ce3 | ||
|
|
f3afa4d9d2 | ||
|
|
e33d81eda1 | ||
|
|
0fc4863dc4 | ||
|
|
63471a5533 | ||
|
|
050e90cb7e | ||
|
|
b4643f2e45 | ||
|
|
2d740a76b7 | ||
|
|
0de9e3a02d | ||
|
|
bedaa37e08 | ||
|
|
ef9fa80b76 | ||
|
|
dc2ea625ec | ||
|
|
17df2c188a | ||
|
|
6e050bccdf | ||
|
|
8afbf3ce91 | ||
|
|
9f57bf46a1 | ||
|
|
4ce47da521 | ||
|
|
4dc6382402 | ||
|
|
cb8c1e32d9 | ||
|
|
bf534be128 | ||
|
|
a89c117612 | ||
|
|
501dbcd76b | ||
|
|
c0ceff1eec | ||
|
|
6039ac5d4f | ||
|
|
48e87f3c47 | ||
|
|
c7866a1270 |
@@ -1,12 +1,7 @@
|
|||||||
FROM elixir:1.16-otp-25
|
FROM elixir:1.17-otp-27
|
||||||
|
|
||||||
RUN apt update -yq
|
RUN apt install -yq curl gnupg
|
||||||
RUN apt install -yq curl gnupg mc inotify-tools
|
|
||||||
RUN apt --fix-broken install
|
RUN apt --fix-broken install
|
||||||
RUN apt remove -y nodejs nodejs-doc
|
|
||||||
RUN curl -sL https://deb.nodesource.com/setup_18.x | bash -
|
|
||||||
RUN apt install -y nodejs
|
|
||||||
RUN npm install --global yarn
|
|
||||||
|
|
||||||
RUN mix local.hex --force
|
RUN mix local.hex --force
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,7 @@ version: "0.1"
|
|||||||
|
|
||||||
services:
|
services:
|
||||||
db:
|
db:
|
||||||
image: postgres:14.3
|
image: postgres:13-alpine
|
||||||
restart: always
|
restart: always
|
||||||
environment:
|
environment:
|
||||||
POSTGRES_USER: postgres
|
POSTGRES_USER: postgres
|
||||||
@@ -10,13 +10,13 @@ services:
|
|||||||
ports:
|
ports:
|
||||||
- "5432:5432"
|
- "5432:5432"
|
||||||
volumes:
|
volumes:
|
||||||
- db:/var/lib/postgresql/data
|
- db-new:/var/lib/postgresql/data
|
||||||
|
|
||||||
wanderer:
|
wanderer:
|
||||||
environment:
|
environment:
|
||||||
PORT: 8000
|
PORT: 8000
|
||||||
DB_HOST: db
|
DB_HOST: db
|
||||||
WEB_APP_URL: "http://localhost:4444"
|
WEB_APP_URL: "http://localhost:8000"
|
||||||
ERL_AFLAGS: "-kernel shell_history enabled"
|
ERL_AFLAGS: "-kernel shell_history enabled"
|
||||||
build:
|
build:
|
||||||
context: .
|
context: .
|
||||||
@@ -33,4 +33,4 @@ services:
|
|||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
elixir-artifacts: {}
|
elixir-artifacts: {}
|
||||||
db: {}
|
db-new: {}
|
||||||
|
|||||||
15
.github/FUNDING.yml
vendored
Normal file
15
.github/FUNDING.yml
vendored
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# These are supported funding model platforms
|
||||||
|
|
||||||
|
github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2]
|
||||||
|
patreon: WandererLtd
|
||||||
|
open_collective: # Replace with a single Open Collective username
|
||||||
|
ko_fi: # Replace with a single Ko-fi username
|
||||||
|
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
|
||||||
|
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
|
||||||
|
liberapay: # Replace with a single Liberapay username
|
||||||
|
issuehunt: # Replace with a single IssueHunt username
|
||||||
|
lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry
|
||||||
|
polar: # Replace with a single Polar username
|
||||||
|
buy_me_a_coffee: # Replace with a single Buy Me a Coffee username
|
||||||
|
thanks_dev: # Replace with a single thanks.dev username
|
||||||
|
custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2']
|
||||||
32
.github/workflows/build.yml
vendored
32
.github/workflows/build.yml
vendored
@@ -58,6 +58,8 @@ jobs:
|
|||||||
otp: ["27"]
|
otp: ["27"]
|
||||||
elixir: ["1.17"]
|
elixir: ["1.17"]
|
||||||
node-version: ["18.x"]
|
node-version: ["18.x"]
|
||||||
|
outputs:
|
||||||
|
commit_hash: ${{ steps.generate-changelog.outputs.commit_hash }}
|
||||||
steps:
|
steps:
|
||||||
- name: Prepare
|
- name: Prepare
|
||||||
run: |
|
run: |
|
||||||
@@ -108,16 +110,17 @@ jobs:
|
|||||||
run: mix compile
|
run: mix compile
|
||||||
|
|
||||||
- name: Generate Changelog & Update Tag Version
|
- name: Generate Changelog & Update Tag Version
|
||||||
|
id: generate-changelog
|
||||||
run: |
|
run: |
|
||||||
git config --global user.name 'CI'
|
git config --global user.name 'CI'
|
||||||
git config --global user.email 'ci@users.noreply.github.com'
|
git config --global user.email 'ci@users.noreply.github.com'
|
||||||
mix git_ops.release --force-patch --yes
|
mix git_ops.release --force-patch --yes
|
||||||
git push --follow-tags
|
git push --follow-tags
|
||||||
|
echo "commit_hash=$(git rev-parse HEAD)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
docker:
|
docker:
|
||||||
name: 🛠 Build Docker Images
|
name: 🛠 Build Docker Images
|
||||||
needs:
|
needs: build
|
||||||
- build
|
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
permissions:
|
permissions:
|
||||||
checks: write
|
checks: write
|
||||||
@@ -141,6 +144,7 @@ jobs:
|
|||||||
- name: ⬇️ Checkout repo
|
- name: ⬇️ Checkout repo
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
|
ref: ${{ needs.build.outputs.commit_hash }}
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Prepare Changelog
|
- name: Prepare Changelog
|
||||||
@@ -189,6 +193,30 @@ jobs:
|
|||||||
- name: Image digest
|
- name: Image digest
|
||||||
run: echo ${{ steps.build.outputs.digest }}
|
run: echo ${{ steps.build.outputs.digest }}
|
||||||
|
|
||||||
|
- uses: markpatterson27/markdown-to-output@v1
|
||||||
|
id: extract-changelog
|
||||||
|
with:
|
||||||
|
filepath: CHANGELOG.md
|
||||||
|
|
||||||
|
- name: Get content
|
||||||
|
uses: 2428392/gh-truncate-string-action@v1.3.0
|
||||||
|
id: get-content
|
||||||
|
with:
|
||||||
|
stringToTruncate: |
|
||||||
|
📣 Wanderer new release available 🎉
|
||||||
|
|
||||||
|
**Version**: ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
|
|
||||||
|
${{ steps.extract-changelog.outputs.body }}
|
||||||
|
maxLength: 500
|
||||||
|
truncationSymbol: "…"
|
||||||
|
|
||||||
|
- name: Discord Webhook Action
|
||||||
|
uses: tsickert/discord-webhook@v5.3.0
|
||||||
|
with:
|
||||||
|
webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
|
||||||
|
content: ${{ steps.get-content.outputs.string }}
|
||||||
|
|
||||||
create-release:
|
create-release:
|
||||||
name: 🏷 Create Release
|
name: 🏷 Create Release
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
|
|||||||
2
.gitignore
vendored
2
.gitignore
vendored
@@ -12,6 +12,8 @@
|
|||||||
|
|
||||||
# Ignore assets that are produced by build tools.
|
# Ignore assets that are produced by build tools.
|
||||||
/priv/static/assets/
|
/priv/static/assets/
|
||||||
|
/priv/static/icons/
|
||||||
|
/priv/static/images/
|
||||||
/priv/static/*.js
|
/priv/static/*.js
|
||||||
/priv/static/*.css
|
/priv/static/*.css
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
erlang 27.0.1
|
erlang 25.3
|
||||||
elixir 1.17-otp-27
|
elixir 1.16-otp-25
|
||||||
nodejs 18.0.0
|
nodejs 18.0.0
|
||||||
|
|||||||
341
CHANGELOG.md
341
CHANGELOG.md
@@ -2,91 +2,328 @@
|
|||||||
|
|
||||||
<!-- changelog -->
|
<!-- changelog -->
|
||||||
|
|
||||||
## [v1.0.13](https://github.com/wanderer-industries/wanderer/compare/v1.0.12...v1.0.13) (2024-09-21)
|
## [v1.15.1](https://github.com/wanderer-industries/wanderer/compare/v1.15.0...v1.15.1) (2024-11-07)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes:
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Dev: Update .devcontainer instructions
|
||||||
|
|
||||||
|
## [v1.15.0](https://github.com/wanderer-industries/wanderer/compare/v1.14.1...v1.15.0) (2024-11-07)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Connections: Add connection mark EOL time (#56)
|
||||||
|
|
||||||
|
## [v1.14.1](https://github.com/wanderer-industries/wanderer/compare/v1.14.0...v1.14.1) (2024-11-06)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Core: Fix character tracking permissions
|
||||||
|
|
||||||
|
## [v1.14.0](https://github.com/wanderer-industries/wanderer/compare/v1.13.12...v1.14.0) (2024-11-05)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* ACL: Add an ability to assign member role without DnD
|
||||||
|
|
||||||
|
## [v1.13.12](https://github.com/wanderer-industries/wanderer/compare/v1.13.11...v1.13.12) (2024-11-04)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix system revert issues
|
||||||
|
|
||||||
|
## [v1.13.11](https://github.com/wanderer-industries/wanderer/compare/v1.13.10...v1.13.11) (2024-11-02)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix system revert issues
|
||||||
|
|
||||||
|
## [v1.13.10](https://github.com/wanderer-industries/wanderer/compare/v1.13.9...v1.13.10) (2024-11-01)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix system revert issues
|
||||||
|
|
||||||
|
## [v1.13.9](https://github.com/wanderer-industries/wanderer/compare/v1.13.8...v1.13.9) (2024-11-01)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## [v1.13.8](https://github.com/wanderer-industries/wanderer/compare/v1.13.7...v1.13.8) (2024-10-28)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## [v1.13.0](https://github.com/wanderer-industries/wanderer/compare/v1.12.11...v1.13.0) (2024-10-28)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Core: Use ESI /characters/affiliation API
|
||||||
|
|
||||||
|
## [v1.12.4](https://github.com/wanderer-industries/wanderer/compare/v1.12.3...v1.12.4) (2024-10-21)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix systems cleanup
|
||||||
|
|
||||||
|
## [v1.12.3](https://github.com/wanderer-industries/wanderer/compare/v1.12.2...v1.12.3) (2024-10-18)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix regression issues
|
||||||
|
|
||||||
|
## [v1.12.1](https://github.com/wanderer-industries/wanderer/compare/v1.12.0...v1.12.1) (2024-10-16)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix system add error after map page refresh
|
||||||
|
|
||||||
|
## [v1.12.0](https://github.com/wanderer-industries/wanderer/compare/v1.11.5...v1.12.0) (2024-10-16)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Map: Prettify user settings
|
||||||
|
|
||||||
|
## [v1.11.0](https://github.com/wanderer-industries/wanderer/compare/v1.10.0...v1.11.0) (2024-10-14)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Map: Add map level option to store custom labels
|
||||||
|
|
||||||
|
## [v1.10.0](https://github.com/wanderer-industries/wanderer/compare/v1.9.0...v1.10.0) (2024-10-13)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Map: Link signature on splash
|
||||||
|
|
||||||
|
## [v1.5.0](https://github.com/wanderer-industries/wanderer/compare/v1.4.0...v1.5.0) (2024-10-11)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Map: Follow Character on Map and auto select their current system
|
||||||
|
|
||||||
|
## [v1.3.6](https://github.com/wanderer-industries/wanderer/compare/v1.3.5...v1.3.6) (2024-10-09)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Signatures: Signatures update fixes
|
||||||
|
|
||||||
|
## [v1.3.0](https://github.com/wanderer-industries/wanderer/compare/v1.2.10...v1.3.0) (2024-10-07)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Map: Fix default sort
|
||||||
|
|
||||||
|
* Map: Remove resizible and fix styles of column sorting
|
||||||
|
|
||||||
|
* Map: Revision of sorting from also adding ability to sort all columns
|
||||||
|
|
||||||
|
## [v1.2.6](https://github.com/wanderer-industries/wanderer/compare/v1.2.5...v1.2.6) (2024-10-05)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Core: Stability & performance improvements
|
||||||
|
|
||||||
|
## [v1.2.5](https://github.com/wanderer-industries/wanderer/compare/v1.2.4...v1.2.5) (2024-10-04)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Core: Add system "true security" correction
|
||||||
|
|
||||||
|
## [v1.2.4](https://github.com/wanderer-industries/wanderer/compare/v1.2.3...v1.2.4) (2024-10-03)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Remove duplicate connections
|
||||||
|
|
||||||
|
## [v1.2.3](https://github.com/wanderer-industries/wanderer/compare/v1.2.2...v1.2.3) (2024-10-02)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix map loading after select a different map.
|
||||||
|
|
||||||
|
## [v1.2.1](https://github.com/wanderer-industries/wanderer/compare/v1.2.0...v1.2.1) (2024-10-02)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* ACL: Fix allowing to save map/access list with empty owner set
|
||||||
|
|
||||||
|
## [v1.2.0](https://github.com/wanderer-industries/wanderer/compare/v1.1.0...v1.2.0) (2024-09-29)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Map: Add ability to open jump planner from routes
|
||||||
|
|
||||||
|
## [v1.1.0](https://github.com/wanderer-industries/wanderer/compare/v1.0.23...v1.1.0) (2024-09-29)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Map: Add highlighting for imperial space systems depends on faction
|
||||||
|
|
||||||
|
## [v1.0.23](https://github.com/wanderer-industries/wanderer/compare/v1.0.22...v1.0.23) (2024-09-25)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Main map doesn't load back after refreshing/switching pages
|
||||||
|
|
||||||
|
## [v1.0.22](https://github.com/wanderer-industries/wanderer/compare/v1.0.21...v1.0.22) (2024-09-25)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* Map: Main map doesn't load back after refreshing/switching pages
|
||||||
|
|
||||||
|
## [v1.0.21](https://github.com/wanderer-industries/wanderer/compare/v1.0.20...v1.0.21) (2024-09-24)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* Map: Main map doesn't load back after refreshing/switching pages
|
||||||
|
|
||||||
|
## [v1.0.20](https://github.com/wanderer-industries/wanderer/compare/v1.0.19...v1.0.20) (2024-09-23)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* core: Small fixes & improvements
|
||||||
|
|
||||||
|
## [v1.0.19](https://github.com/wanderer-industries/wanderer/compare/v1.0.18...v1.0.19) (2024-09-23)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* ACL: Fix adding empty members list
|
||||||
|
|
||||||
|
## [v1.0.18](https://github.com/wanderer-industries/wanderer/compare/v1.0.17...v1.0.18) (2024-09-22)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* ACL: Cant delete ACL list after map deletion #5
|
||||||
|
|
||||||
|
## [v1.0.16](https://github.com/wanderer-industries/wanderer/compare/v1.0.15...v1.0.16) (2024-09-21)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* Map: commented console log
|
||||||
|
|
||||||
|
* Map: add console log for check sys loading
|
||||||
|
|
||||||
|
* Map: add key for cache changes detecting
|
||||||
|
|
||||||
|
## [v1.0.15](https://github.com/wanderer-industries/wanderer/compare/v1.0.14...v1.0.15) (2024-09-21)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
* map: Show a proper user notification if map was deleted/archived
|
||||||
|
|
||||||
|
## [v1.0.14](https://github.com/wanderer-industries/wanderer/compare/v1.0.13...v1.0.14) (2024-09-21)
|
||||||
|
|
||||||
|
## [v1.0.13](https://github.com/wanderer-industries/wanderer/compare/v1.0.12...v1.0.13) (2024-09-21)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
* tracking: Ensure user has at least one character tracked to work with map
|
* tracking: Ensure user has at least one character tracked to work with map
|
||||||
|
|
||||||
## [v1.0.12](https://github.com/wanderer-industries/wanderer/compare/v1.0.11...v1.0.12) (2024-09-20)
|
## [v1.0.12](https://github.com/wanderer-industries/wanderer/compare/v1.0.11...v1.0.12) (2024-09-20)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes:
|
|
||||||
|
|
||||||
* audit: Hide character for non-character map activities
|
* audit: Hide character for non-character map activities
|
||||||
|
|
||||||
## [v1.0.11](https://github.com/wanderer-industries/wanderer/compare/v1.0.10...v1.0.11) (2024-09-20)
|
## [v1.0.11](https://github.com/wanderer-industries/wanderer/compare/v1.0.10...v1.0.11) (2024-09-20)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.10](https://github.com/wanderer-industries/wanderer/compare/v1.0.9...v1.0.10) (2024-09-19)
|
## [v1.0.10](https://github.com/wanderer-industries/wanderer/compare/v1.0.9...v1.0.10) (2024-09-19)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes:
|
|
||||||
|
|
||||||
* signatures: Fix update signatures error if no character tracked on map
|
* signatures: Fix update signatures error if no character tracked on map
|
||||||
|
|
||||||
## [v1.0.9](https://github.com/wanderer-industries/wanderer/compare/v1.0.8...v1.0.9) (2024-09-19)
|
## [v1.0.9](https://github.com/wanderer-industries/wanderer/compare/v1.0.8...v1.0.9) (2024-09-19)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes:
|
|
||||||
|
|
||||||
* core: Fix system add error if it's already added on map
|
* core: Fix system add error if it's already added on map
|
||||||
|
|
||||||
## [v1.0.8](https://github.com/wanderer-industries/wanderer/compare/v1.0.7...v1.0.8) (2024-09-19)
|
## [v1.0.8](https://github.com/wanderer-industries/wanderer/compare/v1.0.7...v1.0.8) (2024-09-19)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes:
|
|
||||||
|
|
||||||
* docker: Fix DB connection in docker-compose internal network
|
* docker: Fix DB connection in docker-compose internal network
|
||||||
|
|
||||||
## [v1.0.7](https://github.com/wanderer-industries/wanderer/compare/v1.0.6...v1.0.7) (2024-09-19)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.6](https://github.com/wanderer-industries/wanderer/compare/v1.0.5...v1.0.6) (2024-09-18)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.5](https://github.com/wanderer-industries/wanderer/compare/v1.0.4...v1.0.5) (2024-09-18)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.4](https://github.com/wanderer-industries/wanderer/compare/v1.0.3...v1.0.4) (2024-09-18)
|
## [v1.0.4](https://github.com/wanderer-industries/wanderer/compare/v1.0.3...v1.0.4) (2024-09-18)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
|
||||||
### Bug Fixes:
|
|
||||||
|
|
||||||
* core: skip search results for failed character info request
|
* core: skip search results for failed character info request
|
||||||
|
|
||||||
## [v1.0.3](https://github.com/wanderer-industries/wanderer/compare/v1.0.2...v1.0.3) (2024-09-18)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.2](https://github.com/wanderer-industries/wanderer/compare/v1.0.1...v1.0.2) (2024-09-18)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
## [v1.0.1](https://github.com/wanderer-industries/wanderer/compare/v1.0.0...v1.0.1) (2024-09-18)
|
## [v1.0.1](https://github.com/wanderer-industries/wanderer/compare/v1.0.0...v1.0.1) (2024-09-18)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -54,7 +54,14 @@ Now you can visit [`localhost:8000`](http://localhost:8000) from your browser.
|
|||||||
#### Using .devcontainer
|
#### Using .devcontainer
|
||||||
|
|
||||||
- Run devcontainer
|
- Run devcontainer
|
||||||
- See how to start server in #setup section
|
- Install additional dependencies inside Dev container
|
||||||
|
|
||||||
|
`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# mix setup`
|
||||||
|
|
||||||
|
- See how to run server in #Run section
|
||||||
|
|
||||||
#### Using nix flakes
|
#### Using nix flakes
|
||||||
|
|
||||||
|
|||||||
@@ -28,6 +28,12 @@ body {
|
|||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#bg-canvas {
|
||||||
|
position: absolute;
|
||||||
|
width: 100vw;
|
||||||
|
height: 100vh;
|
||||||
|
}
|
||||||
|
|
||||||
.ccp-font {
|
.ccp-font {
|
||||||
font-family: 'Shentox', 'Rogan', sans-serif !important;
|
font-family: 'Shentox', 'Rogan', sans-serif !important;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -67,3 +67,44 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.p-sortable-column {
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: bold;
|
||||||
|
padding: 3px 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-selectable-row td {
|
||||||
|
padding: 4px 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-sortable-column > .p-column-header-content > span:last-child {
|
||||||
|
transform: scale(0.7);
|
||||||
|
|
||||||
|
& > svg {
|
||||||
|
margin-left: 4px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dropdown-label, .p-inputtext {
|
||||||
|
padding: 0.25rem 0.75rem;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dropdown-item {
|
||||||
|
padding: 0.25rem 0.5rem;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dropdown-item-group {
|
||||||
|
padding: 0.25rem 0.75rem;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dropdown-trigger {
|
||||||
|
width: 14px;
|
||||||
|
margin: 0 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dropdown-empty-message {
|
||||||
|
padding: 0.25rem 0.5rem;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/* Основной класс диалога */
|
/* Основной класс диалога */
|
||||||
.p-dialog {
|
body .p-dialog {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
//position: absolute;
|
//position: absolute;
|
||||||
@@ -7,11 +7,26 @@
|
|||||||
left: 0;
|
left: 0;
|
||||||
//visibility: hidden;
|
//visibility: hidden;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
border-radius: 6px;
|
border-radius: 2px;
|
||||||
box-shadow: 0 2px 10px 0 rgba(0,0,0,0.2);
|
box-shadow: 0 2px 10px 0 rgba(0,0,0,0.2);
|
||||||
transition: box-shadow 0.3s;
|
transition: box-shadow 0.3s;
|
||||||
background: #fff;
|
|
||||||
z-index: 1000;
|
z-index: 1000;
|
||||||
|
border: 1px solid #212121;
|
||||||
|
background: var(--surface-h);
|
||||||
|
color: var(--text-color);
|
||||||
|
|
||||||
|
.p-dialog-header {
|
||||||
|
background: #171717 !important;
|
||||||
|
color: var(--text-color);
|
||||||
|
|
||||||
|
.p-dialog-header-icon:focus-visible {
|
||||||
|
box-shadow: none !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dialog-footer {
|
||||||
|
border-top: 1px solid var(--surface-d);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Стиль видимого диалога */
|
/* Стиль видимого диалога */
|
||||||
@@ -45,12 +60,12 @@
|
|||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
background: #f4f4f4;
|
background: #f4f4f4;
|
||||||
border-bottom: 1px solid #ddd;
|
//border-bottom: 1px solid #ddd;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Содержимое диалога */
|
/* Содержимое диалога */
|
||||||
.p-dialog-content {
|
.p-dialog-content {
|
||||||
padding: 1rem;
|
padding: 0.5rem;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
flex: 1;
|
flex: 1;
|
||||||
}
|
}
|
||||||
@@ -78,23 +93,3 @@
|
|||||||
.p-dialog-header-close .pi {
|
.p-dialog-header-close .pi {
|
||||||
font-size: 1.25rem;
|
font-size: 1.25rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Тема Saga Blue (пример) */
|
|
||||||
body .p-dialog {
|
|
||||||
background: var(--surface-a);
|
|
||||||
color: var(--text-color);
|
|
||||||
}
|
|
||||||
|
|
||||||
body .p-dialog .p-dialog-header,
|
|
||||||
body .p-dialog .p-dialog-footer {
|
|
||||||
background: var(--surface-b);
|
|
||||||
color: var(--text-color);
|
|
||||||
}
|
|
||||||
|
|
||||||
body .p-dialog .p-dialog-header {
|
|
||||||
border-bottom: 1px solid var(--surface-d);
|
|
||||||
}
|
|
||||||
|
|
||||||
body .p-dialog .p-dialog-footer {
|
|
||||||
border-top: 1px solid var(--surface-d);
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -9,6 +9,7 @@
|
|||||||
--surface-d: #3f4b5b;
|
--surface-d: #3f4b5b;
|
||||||
--surface-e: #2a323d;
|
--surface-e: #2a323d;
|
||||||
--surface-f: #2a323d;
|
--surface-f: #2a323d;
|
||||||
|
--surface-h: #171717;
|
||||||
--text-color: rgba(255, 255, 255, 0.87);
|
--text-color: rgba(255, 255, 255, 0.87);
|
||||||
--text-color-secondary: rgba(255, 255, 255, 0.6);
|
--text-color-secondary: rgba(255, 255, 255, 0.6);
|
||||||
--primary-color: #8dd0ff;
|
--primary-color: #8dd0ff;
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ const Characters = ({ data }: { data: CharacterTypeRaw[] }) => {
|
|||||||
|
|
||||||
const handleSelect = useCallback(
|
const handleSelect = useCallback(
|
||||||
(character: CharacterTypeRaw) => {
|
(character: CharacterTypeRaw) => {
|
||||||
mapRef.current?.command(Commands.selectSystem, character?.location?.solar_system_id?.toString());
|
mapRef.current?.command(Commands.centerSystem, character?.location?.solar_system_id?.toString());
|
||||||
},
|
},
|
||||||
[mapRef],
|
[mapRef],
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ export const useLabelsMenu = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
// const labels = getLabels(system.labels);
|
// const labels = getLabels(system.labels);
|
||||||
const hasLabels = labels.list.length > 0;
|
const hasLabels = labels?.list?.length > 0;
|
||||||
const statusList = hasLabels ? LABELS_ORDER : LABELS_ORDER.slice(1);
|
const statusList = hasLabels ? LABELS_ORDER : LABELS_ORDER.slice(1);
|
||||||
|
|
||||||
return [
|
return [
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ export const useTagMenu = (
|
|||||||
ref.current = { onSystemTag, systems, systemId };
|
ref.current = { onSystemTag, systems, systemId };
|
||||||
|
|
||||||
return useCallback(() => {
|
return useCallback(() => {
|
||||||
const { onSystemTag, systemId , systems} = ref.current;
|
const { onSystemTag, systemId, systems } = ref.current;
|
||||||
const system = systemId ? getSystemById(systems, systemId) : undefined;
|
const system = systemId ? getSystemById(systems, systemId) : undefined;
|
||||||
|
|
||||||
const isSelectedLetters = AVAILABLE_LETTERS.includes(system?.tag ?? '');
|
const isSelectedLetters = AVAILABLE_LETTERS.includes(system?.tag ?? '');
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import { OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.
|
|||||||
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
|
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
|
|
||||||
interface UseContextMenuSystemHandlersProps {
|
interface UseContextMenuSystemHandlersProps {
|
||||||
hubs: string[];
|
hubs: string[];
|
||||||
@@ -16,8 +17,10 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
|
|||||||
|
|
||||||
const [system, setSystem] = useState<string>();
|
const [system, setSystem] = useState<string>();
|
||||||
|
|
||||||
const ref = useRef({ hubs, system, systems, outCommand });
|
const { deleteSystems } = useDeleteSystems();
|
||||||
ref.current = { hubs, system, systems, outCommand };
|
|
||||||
|
const ref = useRef({ hubs, system, systems, outCommand, deleteSystems });
|
||||||
|
ref.current = { hubs, system, systems, outCommand, deleteSystems };
|
||||||
|
|
||||||
const open = useCallback((ev: any, systemId: string) => {
|
const open = useCallback((ev: any, systemId: string) => {
|
||||||
setSystem(systemId);
|
setSystem(systemId);
|
||||||
@@ -27,12 +30,12 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onDeleteSystem = useCallback(() => {
|
const onDeleteSystem = useCallback(() => {
|
||||||
const { system, outCommand } = ref.current;
|
const { system, deleteSystems } = ref.current;
|
||||||
if (!system) {
|
if (!system) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
outCommand({ type: OutCommand.deleteSystems, data: [system] });
|
deleteSystems([system]);
|
||||||
setSystem(undefined);
|
setSystem(undefined);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
|||||||
@@ -8,17 +8,21 @@ import { getSystemById } from '@/hooks/Mapper/helpers';
|
|||||||
import { useWaypointMenu } from '@/hooks/Mapper/components/contexts/hooks';
|
import { useWaypointMenu } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
import { FastSystemActions } from '@/hooks/Mapper/components/contexts/components';
|
import { FastSystemActions } from '@/hooks/Mapper/components/contexts/components';
|
||||||
|
import { useJumpPlannerMenu } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
|
import { Route } from '@/hooks/Mapper/types/routes.ts';
|
||||||
|
|
||||||
export interface ContextMenuSystemInfoProps {
|
export interface ContextMenuSystemInfoProps {
|
||||||
systemStatics: Map<number, SolarSystemStaticInfoRaw>;
|
systemStatics: Map<number, SolarSystemStaticInfoRaw>;
|
||||||
hubs: string[];
|
hubs: string[];
|
||||||
contextMenuRef: RefObject<ContextMenu>;
|
contextMenuRef: RefObject<ContextMenu>;
|
||||||
systemId: string | undefined;
|
systemId: string | undefined;
|
||||||
|
systemIdFrom?: string | undefined;
|
||||||
systems: SolarSystemRawType[];
|
systems: SolarSystemRawType[];
|
||||||
onOpenSettings(): void;
|
onOpenSettings(): void;
|
||||||
onHubToggle(): void;
|
onHubToggle(): void;
|
||||||
onAddSystem(): void;
|
onAddSystem(): void;
|
||||||
onWaypointSet: WaypointSetContextHandler;
|
onWaypointSet: WaypointSetContextHandler;
|
||||||
|
routes: Route[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
|
export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
|
||||||
@@ -30,9 +34,12 @@ export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
|
|||||||
onAddSystem,
|
onAddSystem,
|
||||||
onWaypointSet,
|
onWaypointSet,
|
||||||
systemId,
|
systemId,
|
||||||
|
systemIdFrom,
|
||||||
hubs,
|
hubs,
|
||||||
|
routes,
|
||||||
}) => {
|
}) => {
|
||||||
const getWaypointMenu = useWaypointMenu(onWaypointSet);
|
const getWaypointMenu = useWaypointMenu(onWaypointSet);
|
||||||
|
const getJumpPlannerMenu = useJumpPlannerMenu(systems, systemIdFrom);
|
||||||
|
|
||||||
const items: MenuItem[] = useMemo(() => {
|
const items: MenuItem[] = useMemo(() => {
|
||||||
const system = systemId ? systemStatics.get(parseInt(systemId)) : undefined;
|
const system = systemId ? systemStatics.get(parseInt(systemId)) : undefined;
|
||||||
@@ -55,7 +62,9 @@ export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
{ separator: true },
|
{ separator: true },
|
||||||
|
...getJumpPlannerMenu(system, routes),
|
||||||
...getWaypointMenu(systemId, system.system_class),
|
...getWaypointMenu(systemId, system.system_class),
|
||||||
{
|
{
|
||||||
label: !hubs.includes(systemId) ? 'Add in Routes' : 'Remove from Routes',
|
label: !hubs.includes(systemId) ? 'Add in Routes' : 'Remove from Routes',
|
||||||
@@ -72,7 +81,17 @@ export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
|
|||||||
]
|
]
|
||||||
: []),
|
: []),
|
||||||
];
|
];
|
||||||
}, [systemId, systemStatics, systems, getWaypointMenu, hubs, onHubToggle, onAddSystem, onOpenSettings]);
|
}, [
|
||||||
|
systemId,
|
||||||
|
systemStatics,
|
||||||
|
systems,
|
||||||
|
getJumpPlannerMenu,
|
||||||
|
getWaypointMenu,
|
||||||
|
hubs,
|
||||||
|
onHubToggle,
|
||||||
|
onAddSystem,
|
||||||
|
onOpenSettings,
|
||||||
|
]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import { Commands, MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Ma
|
|||||||
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
|
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
interface UseContextMenuSystemHandlersProps {
|
interface UseContextMenuSystemHandlersProps {
|
||||||
hubs: string[];
|
hubs: string[];
|
||||||
@@ -15,16 +16,21 @@ export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand, mapRef }: U
|
|||||||
const contextMenuRef = useRef<ContextMenu | null>(null);
|
const contextMenuRef = useRef<ContextMenu | null>(null);
|
||||||
|
|
||||||
const [system, setSystem] = useState<string>();
|
const [system, setSystem] = useState<string>();
|
||||||
|
const routeRef = useRef<(SolarSystemStaticInfoRaw | undefined)[]>([]);
|
||||||
|
|
||||||
const ref = useRef({ hubs, system, outCommand, mapRef });
|
const ref = useRef({ hubs, system, outCommand, mapRef });
|
||||||
ref.current = { hubs, system, outCommand, mapRef };
|
ref.current = { hubs, system, outCommand, mapRef };
|
||||||
|
|
||||||
const open = useCallback((ev: React.SyntheticEvent, systemId: string) => {
|
const open = useCallback(
|
||||||
setSystem(systemId);
|
(ev: React.SyntheticEvent, systemId: string, route: (SolarSystemStaticInfoRaw | undefined)[]) => {
|
||||||
ev.preventDefault();
|
setSystem(systemId);
|
||||||
ctxManager.next('ctxSysInfo', contextMenuRef.current);
|
routeRef.current = route;
|
||||||
contextMenuRef.current?.show(ev);
|
ev.preventDefault();
|
||||||
}, []);
|
ctxManager.next('ctxSysInfo', contextMenuRef.current);
|
||||||
|
contextMenuRef.current?.show(ev);
|
||||||
|
},
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
|
||||||
const onHubToggle = useCallback(() => {
|
const onHubToggle = useCallback(() => {
|
||||||
const { hubs, system, outCommand } = ref.current;
|
const { hubs, system, outCommand } = ref.current;
|
||||||
@@ -42,19 +48,19 @@ export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand, mapRef }: U
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onAddSystem = useCallback(() => {
|
const onAddSystem = useCallback(() => {
|
||||||
const { system, outCommand, mapRef } = ref.current;
|
const { system: solarSystemId, outCommand, mapRef } = ref.current;
|
||||||
if (!system) {
|
if (!solarSystemId) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
outCommand({
|
outCommand({
|
||||||
type: OutCommand.addSystem,
|
type: OutCommand.addSystem,
|
||||||
data: {
|
data: {
|
||||||
system_id: system,
|
system_id: solarSystemId,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
mapRef.current?.command(Commands.selectSystem, system);
|
mapRef.current?.command(Commands.centerSystem, solarSystemId);
|
||||||
setSystem(undefined);
|
setSystem(undefined);
|
||||||
}, 200);
|
}, 200);
|
||||||
}, []);
|
}, []);
|
||||||
|
|||||||
@@ -1,17 +1,17 @@
|
|||||||
import { Node } from 'reactflow';
|
import { Node } from 'reactflow';
|
||||||
import { useRef, useState } from 'react';
|
import { useCallback, useRef, useState } from 'react';
|
||||||
import { ContextMenu } from 'primereact/contextmenu';
|
import { ContextMenu } from 'primereact/contextmenu';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
|
||||||
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
|
||||||
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
|
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
|
|
||||||
export const useContextMenuSystemMultipleHandlers = () => {
|
export const useContextMenuSystemMultipleHandlers = () => {
|
||||||
const contextMenuRef = useRef<ContextMenu | null>(null);
|
const contextMenuRef = useRef<ContextMenu | null>(null);
|
||||||
const { outCommand } = useMapRootState();
|
|
||||||
const [systems, setSystems] = useState<Node<SolarSystemRawType>[]>();
|
const [systems, setSystems] = useState<Node<SolarSystemRawType>[]>();
|
||||||
|
|
||||||
|
const { deleteSystems } = useDeleteSystems();
|
||||||
|
|
||||||
const handleSystemMultipleContext: NodeSelectionMouseHandler = (ev, systems_) => {
|
const handleSystemMultipleContext: NodeSelectionMouseHandler = (ev, systems_) => {
|
||||||
setSystems(systems_);
|
setSystems(systems_);
|
||||||
ev.preventDefault();
|
ev.preventDefault();
|
||||||
@@ -19,7 +19,7 @@ export const useContextMenuSystemMultipleHandlers = () => {
|
|||||||
contextMenuRef.current?.show(ev);
|
contextMenuRef.current?.show(ev);
|
||||||
};
|
};
|
||||||
|
|
||||||
const onDeleteSystems = () => {
|
const onDeleteSystems = useCallback(() => {
|
||||||
if (!systems) {
|
if (!systems) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -29,12 +29,11 @@ export const useContextMenuSystemMultipleHandlers = () => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
outCommand({ type: OutCommand.deleteSystems, data: sysToDel });
|
deleteSystems(sysToDel);
|
||||||
};
|
}, [deleteSystems, systems]);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
handleSystemMultipleContext,
|
handleSystemMultipleContext,
|
||||||
|
|
||||||
contextMenuRef,
|
contextMenuRef,
|
||||||
onDeleteSystems,
|
onDeleteSystems,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1 +1,3 @@
|
|||||||
export * from './useWaypointMenu';
|
export * from './useWaypointMenu';
|
||||||
|
export * from './useJumpPlannerMenu';
|
||||||
|
export * from './useDeleteSystems';
|
||||||
|
|||||||
@@ -0,0 +1,18 @@
|
|||||||
|
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
|
export const useDeleteSystems = () => {
|
||||||
|
const { outCommand } = useMapRootState();
|
||||||
|
|
||||||
|
const deleteSystems = (systemIds: string[]) => {
|
||||||
|
if (!systemIds || !systemIds.length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
outCommand({ type: OutCommand.deleteSystems, data: systemIds });
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
deleteSystems,
|
||||||
|
};
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './useJumpPlannerMenu.tsx';
|
||||||
@@ -0,0 +1,129 @@
|
|||||||
|
import { MenuItem } from 'primereact/menuitem';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
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';
|
||||||
|
|
||||||
|
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];
|
||||||
|
|
||||||
|
enum JUMP_SHIP_TYPE {
|
||||||
|
BLACK_OPS = 'Marshal',
|
||||||
|
JUMP_FREIGHTER = 'Anshar',
|
||||||
|
RORQUAL = 'Rorqual',
|
||||||
|
CAPITAL = 'Thanatos',
|
||||||
|
SUPER_CAPITAL = 'Avatar',
|
||||||
|
}
|
||||||
|
|
||||||
|
export const openJumpPlan = (jumpShipType: JUMP_SHIP_TYPE, from: string, to: string) => {
|
||||||
|
return window.open(`https://evemaps.dotlan.net/jump/${jumpShipType},544/${from}:${to}`, '_blank');
|
||||||
|
};
|
||||||
|
|
||||||
|
const BRACKET_ICONS = {
|
||||||
|
npcsuperCarrier_32: '/icons/brackets/npcsuperCarrier_32.png',
|
||||||
|
carrier_32: '/icons/brackets/carrier_32.png',
|
||||||
|
battleship_32: '/icons/brackets/battleship_32.png',
|
||||||
|
freighter_32: '/icons/brackets/freighter_32.png',
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderIcon = (icon: string) => {
|
||||||
|
return (
|
||||||
|
<div className="flex justify-center items-center mr-1.5 pt-px">
|
||||||
|
<img src={icon} style={{ width: 20, height: 20 }} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const useJumpPlannerMenu = (
|
||||||
|
systems: SolarSystemRawType[],
|
||||||
|
systemIdFrom?: string | undefined,
|
||||||
|
): ((systemId: SolarSystemStaticInfoRaw, routes: Route[]) => MenuItem[]) => {
|
||||||
|
return useCallback(
|
||||||
|
(destination: SolarSystemStaticInfoRaw) => {
|
||||||
|
if (!destination || !systemIdFrom) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const origin = getSystemById(systems, systemIdFrom)?.system_static_info;
|
||||||
|
|
||||||
|
if (!origin) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
const isShowBOorJumpFreighter =
|
||||||
|
isPossibleSpace(imperialSpace, origin.system_class) && isPossibleSpace(criminalSpace, destination.system_class);
|
||||||
|
|
||||||
|
const isShowCapital =
|
||||||
|
isPossibleSpace(criminalSpace, origin.system_class) && isPossibleSpace(criminalSpace, destination.system_class);
|
||||||
|
|
||||||
|
if (!isShowBOorJumpFreighter && !isShowCapital) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
label: 'In Jump Planner',
|
||||||
|
icon: PrimeIcons.SEND,
|
||||||
|
items: [
|
||||||
|
...(isShowBOorJumpFreighter
|
||||||
|
? [
|
||||||
|
{
|
||||||
|
label: 'Black Ops',
|
||||||
|
icon: renderIcon(BRACKET_ICONS.battleship_32),
|
||||||
|
command: () => {
|
||||||
|
openJumpPlan(JUMP_SHIP_TYPE.BLACK_OPS, origin.solar_system_name, destination.solar_system_name);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Jump Freighter',
|
||||||
|
icon: renderIcon(BRACKET_ICONS.freighter_32),
|
||||||
|
command: () => {
|
||||||
|
openJumpPlan(
|
||||||
|
JUMP_SHIP_TYPE.JUMP_FREIGHTER,
|
||||||
|
origin.solar_system_name,
|
||||||
|
destination.solar_system_name,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Rorqual',
|
||||||
|
icon: renderIcon(BRACKET_ICONS.freighter_32),
|
||||||
|
command: () => {
|
||||||
|
openJumpPlan(JUMP_SHIP_TYPE.RORQUAL, origin.solar_system_name, destination.solar_system_name);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: []),
|
||||||
|
|
||||||
|
...(isShowCapital
|
||||||
|
? [
|
||||||
|
{
|
||||||
|
label: 'Capital',
|
||||||
|
icon: renderIcon(BRACKET_ICONS.carrier_32),
|
||||||
|
command: () => {
|
||||||
|
openJumpPlan(JUMP_SHIP_TYPE.CAPITAL, origin.solar_system_name, destination.solar_system_name);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Super Capital',
|
||||||
|
icon: renderIcon(BRACKET_ICONS.npcsuperCarrier_32),
|
||||||
|
command: () => {
|
||||||
|
openJumpPlan(
|
||||||
|
JUMP_SHIP_TYPE.SUPER_CAPITAL,
|
||||||
|
origin.solar_system_name,
|
||||||
|
destination.solar_system_name,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: []),
|
||||||
|
],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
},
|
||||||
|
[systems, systemIdFrom],
|
||||||
|
);
|
||||||
|
};
|
||||||
2
assets/js/hooks/Mapper/components/hooks/index.ts
Normal file
2
assets/js/hooks/Mapper/components/hooks/index.ts
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
export * from './useSystemInfo';
|
||||||
|
export * from './useGetOwnOnlineCharacters';
|
||||||
33
assets/js/hooks/Mapper/components/hooks/useSystemInfo.ts
Normal file
33
assets/js/hooks/Mapper/components/hooks/useSystemInfo.ts
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
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';
|
||||||
|
|
||||||
|
interface UseSystemInfoProps {
|
||||||
|
systemId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useSystemInfo = ({ systemId }: UseSystemInfoProps) => {
|
||||||
|
const {
|
||||||
|
data: { systems, connections },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const { systems: systemStatics } = useLoadSystemStatic({ systems: [systemId] });
|
||||||
|
|
||||||
|
return useMemo(() => {
|
||||||
|
const staticInfo = systemStatics.get(parseInt(systemId));
|
||||||
|
const dynamicInfo = getSystemById(systems, systemId);
|
||||||
|
|
||||||
|
if (!staticInfo || !dynamicInfo) {
|
||||||
|
throw new Error(`Error on getting system ${systemId}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const leadsTo = connections
|
||||||
|
.filter(x => [x.source, x.target].includes(systemId))
|
||||||
|
.map(x => [x.source, x.target])
|
||||||
|
.flat()
|
||||||
|
.filter(x => x !== systemId);
|
||||||
|
|
||||||
|
return { dynamicInfo, staticInfo, leadsTo };
|
||||||
|
}, [systemStatics, systemId, systems, connections]);
|
||||||
|
};
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import React, { ForwardedRef, forwardRef, MouseEvent, useCallback } from 'react';
|
import { ForwardedRef, forwardRef, MouseEvent, useCallback, useEffect, useRef } from 'react';
|
||||||
import ReactFlow, {
|
import ReactFlow, {
|
||||||
Background,
|
Background,
|
||||||
ConnectionMode,
|
ConnectionMode,
|
||||||
@@ -13,12 +13,15 @@ import ReactFlow, {
|
|||||||
SelectionMode,
|
SelectionMode,
|
||||||
useEdgesState,
|
useEdgesState,
|
||||||
useNodesState,
|
useNodesState,
|
||||||
|
NodeChange,
|
||||||
|
useReactFlow,
|
||||||
} from 'reactflow';
|
} from 'reactflow';
|
||||||
import 'reactflow/dist/style.css';
|
import 'reactflow/dist/style.css';
|
||||||
import classes from './Map.module.scss';
|
import classes from './Map.module.scss';
|
||||||
import './styles/neon-theme.scss';
|
import './styles/neon-theme.scss';
|
||||||
import './styles/eve-common.scss';
|
import './styles/eve-common.scss';
|
||||||
import { MapProvider, useMapState } from './MapProvider';
|
import { MapProvider, useMapState } from './MapProvider';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { useMapHandlers, useUpdateNodes } from './hooks';
|
import { useMapHandlers, useUpdateNodes } from './hooks';
|
||||||
import { MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import {
|
import {
|
||||||
@@ -34,6 +37,7 @@ import { SESSION_KEY } from '@/hooks/Mapper/constants.ts';
|
|||||||
import { SolarSystemConnection, SolarSystemRawType } from '@/hooks/Mapper/types';
|
import { SolarSystemConnection, SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
|
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
|
|
||||||
const DEFAULT_VIEW_PORT = { zoom: 1, x: 0, y: 0 };
|
const DEFAULT_VIEW_PORT = { zoom: 1, x: 0, y: 0 };
|
||||||
|
|
||||||
@@ -94,6 +98,7 @@ interface MapCompProps {
|
|||||||
minimapClasses?: string;
|
minimapClasses?: string;
|
||||||
isShowMinimap?: boolean;
|
isShowMinimap?: boolean;
|
||||||
onSystemContextMenu: (event: MouseEvent<Element>, systemId: string) => void;
|
onSystemContextMenu: (event: MouseEvent<Element>, systemId: string) => void;
|
||||||
|
showKSpaceBG?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const MapComp = ({
|
const MapComp = ({
|
||||||
@@ -105,7 +110,9 @@ const MapComp = ({
|
|||||||
onConnectionInfoClick,
|
onConnectionInfoClick,
|
||||||
onSelectionContextMenu,
|
onSelectionContextMenu,
|
||||||
isShowMinimap,
|
isShowMinimap,
|
||||||
|
showKSpaceBG,
|
||||||
}: MapCompProps) => {
|
}: MapCompProps) => {
|
||||||
|
const { getNode } = useReactFlow();
|
||||||
const [nodes, , onNodesChange] = useNodesState<SolarSystemRawType>(initialNodes);
|
const [nodes, , onNodesChange] = useNodesState<SolarSystemRawType>(initialNodes);
|
||||||
const [edges, , onEdgesChange] = useEdgesState<Edge<SolarSystemConnection>[]>(initialEdges);
|
const [edges, , onEdgesChange] = useEdgesState<Edge<SolarSystemConnection>[]>(initialEdges);
|
||||||
|
|
||||||
@@ -113,8 +120,15 @@ const MapComp = ({
|
|||||||
useUpdateNodes(nodes);
|
useUpdateNodes(nodes);
|
||||||
const { handleRootContext, ...rootCtxProps } = useContextMenuRootHandlers();
|
const { handleRootContext, ...rootCtxProps } = useContextMenuRootHandlers();
|
||||||
const { handleConnectionContext, ...connectionCtxProps } = useContextMenuConnectionHandlers();
|
const { handleConnectionContext, ...connectionCtxProps } = useContextMenuConnectionHandlers();
|
||||||
|
|
||||||
const { update } = useMapState();
|
const { update } = useMapState();
|
||||||
|
const {
|
||||||
|
data: { systems },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const { deleteSystems } = useDeleteSystems();
|
||||||
|
|
||||||
|
const systemsRef = useRef({ systems });
|
||||||
|
systemsRef.current = { systems };
|
||||||
|
|
||||||
const onConnect: OnConnect = useCallback(
|
const onConnect: OnConnect = useCallback(
|
||||||
params => {
|
params => {
|
||||||
@@ -169,13 +183,46 @@ const MapComp = ({
|
|||||||
localStorage.setItem(SESSION_KEY.viewPort, JSON.stringify(viewport));
|
localStorage.setItem(SESSION_KEY.viewPort, JSON.stringify(viewport));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleNodesChange = useCallback(
|
||||||
|
(changes: NodeChange[]) => {
|
||||||
|
const systemsIdsToRemove: string[] = [];
|
||||||
|
const nextChanges = changes.reduce((acc, change) => {
|
||||||
|
if (change.type === 'remove') {
|
||||||
|
const node = getNode(change.id);
|
||||||
|
const { systems = [] } = systemsRef.current;
|
||||||
|
if (node?.data?.id && !systems.map(s => s.id).includes(node?.data?.id)) {
|
||||||
|
return [...acc, change];
|
||||||
|
} else if (!node?.data?.locked) {
|
||||||
|
systemsIdsToRemove.push(node?.data?.id);
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
}
|
||||||
|
return [...acc, change];
|
||||||
|
}, [] as NodeChange[]);
|
||||||
|
|
||||||
|
if (systemsIdsToRemove.length) {
|
||||||
|
deleteSystems(systemsIdsToRemove);
|
||||||
|
}
|
||||||
|
|
||||||
|
onNodesChange(nextChanges);
|
||||||
|
},
|
||||||
|
[deleteSystems, getNode, onNodesChange],
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
update(x => ({
|
||||||
|
...x,
|
||||||
|
showKSpaceBG: showKSpaceBG,
|
||||||
|
}));
|
||||||
|
}, [showKSpaceBG, update]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className={classes.MapRoot}>
|
<div className={classes.MapRoot}>
|
||||||
<ReactFlow
|
<ReactFlow
|
||||||
nodes={nodes}
|
nodes={nodes}
|
||||||
edges={edges}
|
edges={edges}
|
||||||
onNodesChange={onNodesChange}
|
onNodesChange={handleNodesChange}
|
||||||
onEdgesChange={onEdgesChange}
|
onEdgesChange={onEdgesChange}
|
||||||
onConnect={onConnect}
|
onConnect={onConnect}
|
||||||
// TODO we need save into session all of this
|
// TODO we need save into session all of this
|
||||||
@@ -210,10 +257,10 @@ const MapComp = ({
|
|||||||
minZoom={0.2}
|
minZoom={0.2}
|
||||||
maxZoom={1.5}
|
maxZoom={1.5}
|
||||||
elevateNodesOnSelect
|
elevateNodesOnSelect
|
||||||
|
deleteKeyCode={['Delete']}
|
||||||
// TODO need create clear example with problem with that flag
|
// 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)
|
// if system is not visible edge not drawing (and any render in Custom node is not happening)
|
||||||
// onlyRenderVisibleElements
|
// onlyRenderVisibleElements
|
||||||
deleteKeyCode={null}
|
|
||||||
selectionMode={SelectionMode.Partial}
|
selectionMode={SelectionMode.Partial}
|
||||||
>
|
>
|
||||||
{isShowMinimap && <MiniMap pannable zoomable ariaLabel="Mini map" className={minimapClasses} />}
|
{isShowMinimap && <MiniMap pannable zoomable ariaLabel="Mini map" className={minimapClasses} />}
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ export type MapData = MapUnionTypes & {
|
|||||||
isConnecting: boolean;
|
isConnecting: boolean;
|
||||||
hoverNodeId: string | null;
|
hoverNodeId: string | null;
|
||||||
visibleNodes: Set<string>;
|
visibleNodes: Set<string>;
|
||||||
|
showKSpaceBG: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
interface MapProviderProps {
|
interface MapProviderProps {
|
||||||
@@ -27,6 +28,7 @@ const INITIAL_DATA: MapData = {
|
|||||||
connections: [],
|
connections: [],
|
||||||
hoverNodeId: null,
|
hoverNodeId: null,
|
||||||
visibleNodes: new Set(),
|
visibleNodes: new Set(),
|
||||||
|
showKSpaceBG: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export interface MapContextProps {
|
export interface MapContextProps {
|
||||||
@@ -38,6 +40,7 @@ export interface MapContextProps {
|
|||||||
const MapContext = createContext<MapContextProps>({
|
const MapContext = createContext<MapContextProps>({
|
||||||
update: () => {},
|
update: () => {},
|
||||||
data: { ...INITIAL_DATA },
|
data: { ...INITIAL_DATA },
|
||||||
|
// @ts-ignore
|
||||||
outCommand: async () => void 0,
|
outCommand: async () => void 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
@import "@/hooks/Mapper/components/map/styles/eve-common-variables";
|
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
||||||
|
|
||||||
$pastel-blue: #5a7d9a;
|
$pastel-blue: #5a7d9a;
|
||||||
$pastel-pink: #d291bc;
|
$pastel-pink: #d291bc;
|
||||||
@@ -23,6 +23,62 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
border-radius: 5px;
|
border-radius: 5px;
|
||||||
position: relative;
|
position: relative;
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
|
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 {
|
&.selected {
|
||||||
border-color: $pastel-pink;
|
border-color: $pastel-pink;
|
||||||
@@ -39,7 +95,7 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
|
|
||||||
&.eve-system-status-home {
|
&.eve-system-status-home {
|
||||||
border: 1px solid darken($eve-solar-system-status-color-home, 30%);
|
border: 1px solid darken($eve-solar-system-status-color-home, 30%);
|
||||||
background-image: linear-gradient(45deg, $eve-solar-system-status-friendly, transparent);
|
background-image: linear-gradient(275deg, $eve-solar-system-status-friendly, transparent);
|
||||||
|
|
||||||
&.selected {
|
&.selected {
|
||||||
border-color: $eve-solar-system-status-color-home;
|
border-color: $eve-solar-system-status-color-home;
|
||||||
@@ -48,7 +104,7 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
|
|
||||||
&.eve-system-status-friendly {
|
&.eve-system-status-friendly {
|
||||||
border: 1px solid darken($eve-solar-system-status-color-friendly, 20%);
|
border: 1px solid darken($eve-solar-system-status-color-friendly, 20%);
|
||||||
background-image: linear-gradient(45deg, darken($eve-solar-system-status-friendly, 30%), transparent);
|
background-image: linear-gradient(275deg, darken($eve-solar-system-status-friendly, 30%), transparent);
|
||||||
|
|
||||||
&.selected {
|
&.selected {
|
||||||
border-color: darken($eve-solar-system-status-color-friendly, 5%);
|
border-color: darken($eve-solar-system-status-color-friendly, 5%);
|
||||||
@@ -57,7 +113,7 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
|
|
||||||
&.eve-system-status-lookingFor {
|
&.eve-system-status-lookingFor {
|
||||||
border: 1px solid darken($eve-solar-system-status-color-lookingFor, 15%);
|
border: 1px solid darken($eve-solar-system-status-color-lookingFor, 15%);
|
||||||
background-image: linear-gradient(45deg, #45ff8f2f, #457fff2f);
|
background-image: linear-gradient(275deg, #45ff8f2f, #457fff2f);
|
||||||
|
|
||||||
&.selected {
|
&.selected {
|
||||||
border-color: $pastel-pink;
|
border-color: $pastel-pink;
|
||||||
@@ -65,17 +121,16 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-warning {
|
&.eve-system-status-warning {
|
||||||
background-image: linear-gradient(45deg, $eve-solar-system-status-warning, transparent);
|
background-image: linear-gradient(275deg, $eve-solar-system-status-warning, transparent);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-dangerous {
|
&.eve-system-status-dangerous {
|
||||||
background-image: linear-gradient(45deg, $eve-solar-system-status-dangerous, transparent);
|
background-image: linear-gradient(275deg, $eve-solar-system-status-dangerous, transparent);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-target {
|
&.eve-system-status-target {
|
||||||
background-image: linear-gradient(45deg, $eve-solar-system-status-target, transparent);
|
background-image: linear-gradient(275deg, $eve-solar-system-status-target, transparent);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.Bookmarks {
|
.Bookmarks {
|
||||||
@@ -102,7 +157,7 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
//background-color: #833ca4;
|
//background-color: #833ca4;
|
||||||
|
|
||||||
&:not(:first-child) {
|
&:not(:first-child) {
|
||||||
box-shadow: inset 4px -3px 4px rgba(0, 0, 0, .3);
|
box-shadow: inset 4px -3px 4px rgba(0, 0, 0, 0.3);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -125,7 +180,6 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
font-size: 9px;
|
font-size: 9px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.icon {
|
.icon {
|
||||||
@@ -163,9 +217,7 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
}
|
}
|
||||||
|
|
||||||
.solarSystemName {
|
.solarSystemName {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.BottomRow {
|
.BottomRow {
|
||||||
@@ -232,11 +284,19 @@ $tooltip-bg: #202020; // Темный фон для подсказок
|
|||||||
border-color: $pastel-pink;
|
border-color: $pastel-pink;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.HandleTop { top: -2px }
|
&.HandleTop {
|
||||||
|
top: -2px;
|
||||||
|
}
|
||||||
|
|
||||||
&.HandleRight { right: -2px }
|
&.HandleRight {
|
||||||
|
right: -2px;
|
||||||
|
}
|
||||||
|
|
||||||
&.HandleBottom { bottom: -2px }
|
&.HandleBottom {
|
||||||
|
bottom: -2px;
|
||||||
|
}
|
||||||
|
|
||||||
&.HandleLeft { left: -2px }
|
&.HandleLeft {
|
||||||
|
left: -2px;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,9 +19,17 @@ import { PrimeIcons } from 'primereact/api';
|
|||||||
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager.ts';
|
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager.ts';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types';
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick.ts';
|
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[]) => {
|
const sortedLabels = (labels: string[]) => {
|
||||||
if (labels === null) {
|
if (!labels) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -50,6 +58,7 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
|
|||||||
statics,
|
statics,
|
||||||
effect_name,
|
effect_name,
|
||||||
region_name,
|
region_name,
|
||||||
|
region_id,
|
||||||
is_shattered,
|
is_shattered,
|
||||||
solar_system_name,
|
solar_system_name,
|
||||||
} = data.system_static_info;
|
} = data.system_static_info;
|
||||||
@@ -69,6 +78,7 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
|
|||||||
isConnecting,
|
isConnecting,
|
||||||
hoverNodeId,
|
hoverNodeId,
|
||||||
visibleNodes,
|
visibleNodes,
|
||||||
|
showKSpaceBG,
|
||||||
},
|
},
|
||||||
outCommand,
|
outCommand,
|
||||||
} = useMapState();
|
} = useMapState();
|
||||||
@@ -114,13 +124,16 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
|
|||||||
|
|
||||||
const showHandlers = isConnecting || hoverNodeId === id;
|
const showHandlers = isConnecting || hoverNodeId === id;
|
||||||
|
|
||||||
|
const space = showKSpaceBG ? REGIONS_MAP[region_id] : '';
|
||||||
|
const regionClass = showKSpaceBG ? SpaceToClass[space] : null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{visible && (
|
{visible && (
|
||||||
<div className={classes.Bookmarks}>
|
<div className={classes.Bookmarks}>
|
||||||
{labelCustom !== '' && (
|
{labelCustom !== '' && (
|
||||||
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
||||||
<div>{labelCustom}</div>
|
<span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] ">{labelCustom}</span>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
@@ -147,18 +160,24 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<div className={clsx(classes.RootCustomNode, classes[STATUS_CLASSES[status]], { [classes.selected]: selected })}>
|
<div
|
||||||
|
className={clsx(classes.RootCustomNode, regionClass, classes[STATUS_CLASSES[status]], {
|
||||||
|
[classes.selected]: selected,
|
||||||
|
})}
|
||||||
|
>
|
||||||
{visible && (
|
{visible && (
|
||||||
<>
|
<>
|
||||||
<div className={classes.HeadRow}>
|
<div className={classes.HeadRow}>
|
||||||
<div className={clsx(classes.classTitle, classTitleColor)}>{class_title ?? '-'}</div>
|
<div className={clsx(classes.classTitle, classTitleColor, '[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]')}>
|
||||||
|
{class_title ?? '-'}
|
||||||
|
</div>
|
||||||
{tag != null && tag !== '' && (
|
{tag != null && tag !== '' && (
|
||||||
<div className={clsx(classes.TagTitle, 'text-sky-400 font-medium')}>{tag}</div>
|
<div className={clsx(classes.TagTitle, 'text-sky-400 font-medium')}>{tag}</div>
|
||||||
)}
|
)}
|
||||||
<div
|
<div
|
||||||
className={clsx(
|
className={clsx(
|
||||||
classes.classSystemName,
|
classes.classSystemName,
|
||||||
'flex-grow overflow-hidden text-ellipsis whitespace-nowrap font-sans',
|
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] flex-grow overflow-hidden text-ellipsis whitespace-nowrap font-sans',
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
{solar_system_name}
|
{solar_system_name}
|
||||||
@@ -179,11 +198,17 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
|
|||||||
|
|
||||||
<div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
|
<div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
|
||||||
{customName && (
|
{customName && (
|
||||||
<div className="text-blue-300 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">{customName}</div>
|
<div className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] text-blue-300 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
|
||||||
|
{customName}
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{!isWormhole && !customName && (
|
{!isWormhole && !customName && (
|
||||||
<div className="text-stone-400 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
|
<div
|
||||||
|
className={clsx(
|
||||||
|
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] text-stone-300 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5',
|
||||||
|
)}
|
||||||
|
>
|
||||||
{region_name}
|
{region_name}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
@@ -192,10 +217,10 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
|
|||||||
|
|
||||||
<div className="flex items-center justify-end">
|
<div className="flex items-center justify-end">
|
||||||
<div className="flex gap-1 items-center">
|
<div className="flex gap-1 items-center">
|
||||||
{locked && <i className={PrimeIcons.LOCK} style={{ fontSize: '0.45rem' }}></i>}
|
{locked && <i className={PrimeIcons.LOCK} style={{ fontSize: '0.45rem', fontWeight: 'bold' }}></i>}
|
||||||
|
|
||||||
{hubs.includes(solar_system_id.toString()) && (
|
{hubs.includes(solar_system_id.toString()) && (
|
||||||
<i className={PrimeIcons.MAP_MARKER} style={{ fontSize: '0.45rem' }}></i>
|
<i className={PrimeIcons.MAP_MARKER} style={{ fontSize: '0.45rem', fontWeight: 'bold' }}></i>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{charactersInSystem.length > 0 && (
|
{charactersInSystem.length > 0 && (
|
||||||
|
|||||||
@@ -18,5 +18,9 @@ export const WormholeClassComp = ({ id }: WormholeClassComp) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const colorClass = WORMHOLE_CLASS_STYLES[wormholeDataAdditional.wormholeClassID.toString()];
|
const colorClass = WORMHOLE_CLASS_STYLES[wormholeDataAdditional.wormholeClassID.toString()];
|
||||||
return <div className={clsx(colorClass)}>{wormholeDataAdditional.shortName}</div>;
|
return (
|
||||||
|
<div className={clsx(colorClass, '[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]')}>
|
||||||
|
{wormholeDataAdditional.shortName}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -30,11 +30,77 @@ export enum SOLAR_SYSTEM_CLASS_IDS {
|
|||||||
zarzakh = 10100,
|
zarzakh = 10100,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export enum SOLAR_SYSTEM_CLASS_GROUPS {
|
||||||
|
ccp = 'ccp',
|
||||||
|
c1 = 'c1',
|
||||||
|
c2 = 'c2',
|
||||||
|
c3 = 'c3',
|
||||||
|
c4 = 'c4',
|
||||||
|
c5 = 'c5',
|
||||||
|
c6 = 'c6',
|
||||||
|
hs = 'hs',
|
||||||
|
ls = 'ls',
|
||||||
|
ns = 'ns',
|
||||||
|
thera = 'thera',
|
||||||
|
c13 = 'c13',
|
||||||
|
drifter = 'drifter',
|
||||||
|
unknown = 'unknown',
|
||||||
|
pochven = 'pochven',
|
||||||
|
jovian = 'jovian',
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SOLAR_SYSTEM_TO_CLASS_GROUPS_CLASSES = {
|
||||||
|
c1: ['c1'],
|
||||||
|
c2: ['c2'],
|
||||||
|
c3: ['c3'],
|
||||||
|
c4: ['c4'],
|
||||||
|
c5: ['c5'],
|
||||||
|
c6: ['c6'],
|
||||||
|
hs: ['hs'],
|
||||||
|
ls: ['ls'],
|
||||||
|
ns: ['ns'],
|
||||||
|
thera: ['thera'],
|
||||||
|
c13: ['c13'],
|
||||||
|
pochven: ['pochven'],
|
||||||
|
drifter: ['sentinel', 'barbican', 'vidette', 'conflux', 'redoubt'],
|
||||||
|
jove: ['jove'],
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS = {
|
||||||
|
ccp1: SOLAR_SYSTEM_CLASS_GROUPS.ccp,
|
||||||
|
c1: SOLAR_SYSTEM_CLASS_GROUPS.c1,
|
||||||
|
c2: SOLAR_SYSTEM_CLASS_GROUPS.c2,
|
||||||
|
c3: SOLAR_SYSTEM_CLASS_GROUPS.c3,
|
||||||
|
c4: SOLAR_SYSTEM_CLASS_GROUPS.c4,
|
||||||
|
c5: SOLAR_SYSTEM_CLASS_GROUPS.c5,
|
||||||
|
c6: SOLAR_SYSTEM_CLASS_GROUPS.c6,
|
||||||
|
hs: SOLAR_SYSTEM_CLASS_GROUPS.hs,
|
||||||
|
ls: SOLAR_SYSTEM_CLASS_GROUPS.ls,
|
||||||
|
ns: SOLAR_SYSTEM_CLASS_GROUPS.ns,
|
||||||
|
ccp2: SOLAR_SYSTEM_CLASS_GROUPS.ccp,
|
||||||
|
ccp3: SOLAR_SYSTEM_CLASS_GROUPS.ccp,
|
||||||
|
thera: SOLAR_SYSTEM_CLASS_GROUPS.thera,
|
||||||
|
c13: SOLAR_SYSTEM_CLASS_GROUPS.c13,
|
||||||
|
sentinel: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
|
baribican: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
|
vidette: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
|
conflux: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
|
redoubt: SOLAR_SYSTEM_CLASS_GROUPS.drifter,
|
||||||
|
a1: SOLAR_SYSTEM_CLASS_GROUPS.unknown,
|
||||||
|
a2: SOLAR_SYSTEM_CLASS_GROUPS.unknown,
|
||||||
|
a3: SOLAR_SYSTEM_CLASS_GROUPS.unknown,
|
||||||
|
a4: SOLAR_SYSTEM_CLASS_GROUPS.unknown,
|
||||||
|
a5: SOLAR_SYSTEM_CLASS_GROUPS.unknown,
|
||||||
|
ccp4: SOLAR_SYSTEM_CLASS_GROUPS.ccp,
|
||||||
|
pochven: SOLAR_SYSTEM_CLASS_GROUPS.pochven,
|
||||||
|
};
|
||||||
|
|
||||||
type WormholesAdditionalInfoType = {
|
type WormholesAdditionalInfoType = {
|
||||||
id: string;
|
id: string;
|
||||||
shortName: string;
|
shortName: string;
|
||||||
wormholeClassID: number;
|
wormholeClassID: number;
|
||||||
title: string;
|
title: string;
|
||||||
|
shortTitle: string;
|
||||||
effectPower?: number;
|
effectPower?: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -45,6 +111,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
shortName: 'CCP',
|
shortName: 'CCP',
|
||||||
wormholeClassID: -1,
|
wormholeClassID: -1,
|
||||||
title: 'CCP System',
|
title: 'CCP System',
|
||||||
|
shortTitle: 'CCP',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'c1',
|
id: 'c1',
|
||||||
@@ -52,6 +119,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 1,
|
wormholeClassID: 1,
|
||||||
effectPower: 1,
|
effectPower: 1,
|
||||||
title: 'Class 1',
|
title: 'Class 1',
|
||||||
|
shortTitle: 'C1',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'c2',
|
id: 'c2',
|
||||||
@@ -59,6 +127,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 2,
|
wormholeClassID: 2,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 2',
|
title: 'Class 2',
|
||||||
|
shortTitle: 'C2',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'c3',
|
id: 'c3',
|
||||||
@@ -66,6 +135,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 3,
|
wormholeClassID: 3,
|
||||||
effectPower: 3,
|
effectPower: 3,
|
||||||
title: 'Class 3',
|
title: 'Class 3',
|
||||||
|
shortTitle: 'C3',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'c4',
|
id: 'c4',
|
||||||
@@ -73,6 +143,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 4,
|
wormholeClassID: 4,
|
||||||
effectPower: 4,
|
effectPower: 4,
|
||||||
title: 'Class 4',
|
title: 'Class 4',
|
||||||
|
shortTitle: 'C4',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'c5',
|
id: 'c5',
|
||||||
@@ -80,6 +151,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 5,
|
wormholeClassID: 5,
|
||||||
effectPower: 5,
|
effectPower: 5,
|
||||||
title: 'Class 5',
|
title: 'Class 5',
|
||||||
|
shortTitle: 'C5',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'c6',
|
id: 'c6',
|
||||||
@@ -87,42 +159,49 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 6,
|
wormholeClassID: 6,
|
||||||
effectPower: 6,
|
effectPower: 6,
|
||||||
title: 'Class 6',
|
title: 'Class 6',
|
||||||
|
shortTitle: 'C6',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'hs',
|
id: 'hs',
|
||||||
shortName: 'H',
|
shortName: 'H',
|
||||||
wormholeClassID: 7,
|
wormholeClassID: 7,
|
||||||
title: 'High-sec',
|
title: 'High-sec',
|
||||||
|
shortTitle: 'High-sec',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'ls',
|
id: 'ls',
|
||||||
shortName: 'L',
|
shortName: 'L',
|
||||||
wormholeClassID: 8,
|
wormholeClassID: 8,
|
||||||
title: 'Low-sec',
|
title: 'Low-sec',
|
||||||
|
shortTitle: 'Low-sec',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'ns',
|
id: 'ns',
|
||||||
shortName: 'N',
|
shortName: 'N',
|
||||||
wormholeClassID: 9,
|
wormholeClassID: 9,
|
||||||
title: 'Null-sec',
|
title: 'Null-sec',
|
||||||
|
shortTitle: 'Null-sec',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'ccp2',
|
id: 'ccp2',
|
||||||
shortName: 'CCP',
|
shortName: 'CCP',
|
||||||
wormholeClassID: 10,
|
wormholeClassID: 10,
|
||||||
title: 'CCP System',
|
title: 'CCP System',
|
||||||
|
shortTitle: 'CCP',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'ccp3',
|
id: 'ccp3',
|
||||||
shortName: 'CCP',
|
shortName: 'CCP',
|
||||||
wormholeClassID: 11,
|
wormholeClassID: 11,
|
||||||
title: 'CCP System',
|
title: 'CCP System',
|
||||||
|
shortTitle: 'CCP',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'thera',
|
id: 'thera',
|
||||||
shortName: 'T',
|
shortName: 'T',
|
||||||
wormholeClassID: 12,
|
wormholeClassID: 12,
|
||||||
title: 'Class 12 (Thera)',
|
title: 'Class 12 (Thera)',
|
||||||
|
shortTitle: 'Thera',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'c13',
|
id: 'c13',
|
||||||
@@ -130,6 +209,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 13,
|
wormholeClassID: 13,
|
||||||
effectPower: 6,
|
effectPower: 6,
|
||||||
title: 'Class 13 (Shattered Frigate)',
|
title: 'Class 13 (Shattered Frigate)',
|
||||||
|
shortTitle: 'C13',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'sentinel',
|
id: 'sentinel',
|
||||||
@@ -137,6 +217,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 14,
|
wormholeClassID: 14,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 14 (Sentinel Drifter)',
|
title: 'Class 14 (Sentinel Drifter)',
|
||||||
|
shortTitle: 'Sentinel',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'barbican',
|
id: 'barbican',
|
||||||
@@ -144,6 +225,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 15,
|
wormholeClassID: 15,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 15 (Barbican Drifter)',
|
title: 'Class 15 (Barbican Drifter)',
|
||||||
|
shortTitle: 'Barbican',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'vidette',
|
id: 'vidette',
|
||||||
@@ -151,6 +233,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 16,
|
wormholeClassID: 16,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 16 (Vidette Drifter)',
|
title: 'Class 16 (Vidette Drifter)',
|
||||||
|
shortTitle: 'Vidette',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'conflux',
|
id: 'conflux',
|
||||||
@@ -158,6 +241,7 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 17,
|
wormholeClassID: 17,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 17 (Conflux Drifter)',
|
title: 'Class 17 (Conflux Drifter)',
|
||||||
|
shortTitle: 'Conflux',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'redoubt',
|
id: 'redoubt',
|
||||||
@@ -165,59 +249,79 @@ export const WORMHOLES_ADDITIONAL_INFO_RAW: WormholesAdditionalInfoType[] = [
|
|||||||
wormholeClassID: 18,
|
wormholeClassID: 18,
|
||||||
effectPower: 2,
|
effectPower: 2,
|
||||||
title: 'Class 18 (Redoubt Drifter)',
|
title: 'Class 18 (Redoubt Drifter)',
|
||||||
|
shortTitle: 'Redoubt',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'a1',
|
id: 'a1',
|
||||||
shortName: 'A1',
|
shortName: 'A1',
|
||||||
wormholeClassID: 19,
|
wormholeClassID: 19,
|
||||||
title: '(Abyssal class 1)',
|
title: '(Abyssal class 1)',
|
||||||
|
shortTitle: 'A1',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'a2',
|
id: 'a2',
|
||||||
shortName: 'A2',
|
shortName: 'A2',
|
||||||
wormholeClassID: 20,
|
wormholeClassID: 20,
|
||||||
title: '(Abyssal class 2)',
|
title: '(Abyssal class 2)',
|
||||||
|
shortTitle: 'A2',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'a3',
|
id: 'a3',
|
||||||
shortName: 'A3',
|
shortName: 'A3',
|
||||||
wormholeClassID: 21,
|
wormholeClassID: 21,
|
||||||
title: '(Abyssal class 3)',
|
title: '(Abyssal class 3)',
|
||||||
|
shortTitle: 'A3',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'a4',
|
id: 'a4',
|
||||||
shortName: 'A4',
|
shortName: 'A4',
|
||||||
wormholeClassID: 22,
|
wormholeClassID: 22,
|
||||||
title: '(Abyssal class 4)',
|
title: '(Abyssal class 4)',
|
||||||
|
shortTitle: 'A4',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'a5',
|
id: 'a5',
|
||||||
shortName: 'A5',
|
shortName: 'A5',
|
||||||
wormholeClassID: 23,
|
wormholeClassID: 23,
|
||||||
title: '(Abyssal class 5)',
|
title: '(Abyssal class 5)',
|
||||||
|
shortTitle: 'A5',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'ccp4',
|
id: 'ccp4',
|
||||||
shortName: 'CCP',
|
shortName: 'CCP',
|
||||||
wormholeClassID: 24,
|
wormholeClassID: 24,
|
||||||
title: 'CCP System (Penalty)',
|
title: 'CCP System (Penalty)',
|
||||||
|
shortTitle: 'CCP',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'pochven',
|
id: 'pochven',
|
||||||
shortName: 'P',
|
shortName: 'P',
|
||||||
wormholeClassID: 25,
|
wormholeClassID: 25,
|
||||||
title: 'Triglavian space (Pochven)',
|
title: 'Triglavian space (Pochven)',
|
||||||
|
shortTitle: 'Pochven',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: 'zarzakh',
|
id: 'zarzakh',
|
||||||
shortName: 'N',
|
shortName: 'N',
|
||||||
wormholeClassID: 10100,
|
wormholeClassID: 10100,
|
||||||
title: 'Pirate space',
|
title: 'Pirate space',
|
||||||
|
shortTitle: 'Zarzakh',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'k162',
|
||||||
|
shortName: 'K162',
|
||||||
|
wormholeClassID: 10101,
|
||||||
|
title: 'Reverse',
|
||||||
|
shortTitle: 'K162',
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
export const WORMHOLES_ADDITIONAL_INFO: Record<string, WormholesAdditionalInfoType> =
|
export const WORMHOLES_ADDITIONAL_INFO: Record<string, WormholesAdditionalInfoType> =
|
||||||
WORMHOLES_ADDITIONAL_INFO_RAW.reduce((acc, x) => ({ ...acc, [x.id]: x }), {});
|
WORMHOLES_ADDITIONAL_INFO_RAW.reduce((acc, x) => ({ ...acc, [x.id]: x }), {});
|
||||||
|
|
||||||
|
export const WORMHOLES_ADDITIONAL_INFO_BY_CLASS_ID: Record<string, WormholesAdditionalInfoType> =
|
||||||
|
WORMHOLES_ADDITIONAL_INFO_RAW.reduce((acc, x) => ({ ...acc, [x.wormholeClassID]: x }), {});
|
||||||
|
|
||||||
// export const SOLAR_SYSTEM_CLASS_NAMES = {
|
// export const SOLAR_SYSTEM_CLASS_NAMES = {
|
||||||
// ccp1 = ,
|
// ccp1 = ,
|
||||||
// c1 = ,
|
// c1 = ,
|
||||||
|
|||||||
@@ -11,3 +11,7 @@ export const isKnownSpace = (wormholeClassID: number) => {
|
|||||||
|
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const isPossibleSpace = (spaces: number[], wormholeClassID: number) => {
|
||||||
|
return spaces.includes(wormholeClassID);
|
||||||
|
};
|
||||||
|
|||||||
@@ -5,5 +5,6 @@ export * from './useMapRemoveSystems';
|
|||||||
export * from './useCommandsCharacters';
|
export * from './useCommandsCharacters';
|
||||||
export * from './useCommandsConnections';
|
export * from './useCommandsConnections';
|
||||||
export * from './useCommandsConnections';
|
export * from './useCommandsConnections';
|
||||||
|
export * from './useCenterSystem';
|
||||||
export * from './useSelectSystem';
|
export * from './useSelectSystem';
|
||||||
export * from './useMapCommands';
|
export * from './useMapCommands';
|
||||||
|
|||||||
@@ -0,0 +1,18 @@
|
|||||||
|
import { useReactFlow } from 'reactflow';
|
||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
import { CommandCenterSystem } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
|
export const useCenterSystem = () => {
|
||||||
|
const rf = useReactFlow();
|
||||||
|
|
||||||
|
const ref = useRef({ rf });
|
||||||
|
ref.current = { rf };
|
||||||
|
|
||||||
|
return useCallback((systemId: CommandCenterSystem) => {
|
||||||
|
const systemNode = ref.current.rf.getNodes().find(x => x.data.id === systemId);
|
||||||
|
if (!systemNode) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ref.current.rf.setCenter(systemNode.position.x, systemNode.position.y, { duration: 1000 });
|
||||||
|
}, []);
|
||||||
|
};
|
||||||
@@ -2,28 +2,17 @@ import { Node, useReactFlow } from 'reactflow';
|
|||||||
import { useCallback, useRef } from 'react';
|
import { useCallback, useRef } from 'react';
|
||||||
import { CommandAddSystems } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { CommandAddSystems } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { convertSystem2Node } from '../../helpers';
|
import { convertSystem2Node } from '../../helpers';
|
||||||
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
|
||||||
|
|
||||||
export const useMapAddSystems = () => {
|
export const useMapAddSystems = () => {
|
||||||
const rf = useReactFlow();
|
const rf = useReactFlow();
|
||||||
const {
|
|
||||||
data: { systems },
|
|
||||||
update,
|
|
||||||
} = useMapState();
|
|
||||||
|
|
||||||
const ref = useRef({ rf, systems, update });
|
const ref = useRef({ rf });
|
||||||
ref.current = { update, systems, rf };
|
ref.current = { rf };
|
||||||
|
|
||||||
return useCallback(
|
return useCallback((systems: CommandAddSystems) => {
|
||||||
(systems: CommandAddSystems) => {
|
const { rf } = ref.current;
|
||||||
const nodes = rf.getNodes();
|
const nodes = rf.getNodes();
|
||||||
const prepared: Node[] = systems.filter(x => !nodes.some(y => x.id === y.id)).map(convertSystem2Node);
|
const prepared: Node[] = systems.filter(x => !nodes.some(y => x.id === y.id)).map(convertSystem2Node);
|
||||||
rf.addNodes(prepared);
|
rf.addNodes(prepared);
|
||||||
|
}, []);
|
||||||
ref.current.update({
|
|
||||||
systems: [...ref.current.systems.filter(sys => systems.some(x => sys.id !== x.id)), ...systems],
|
|
||||||
});
|
|
||||||
},
|
|
||||||
[rf],
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -4,18 +4,18 @@ import { CommandSelectSystem } from '@/hooks/Mapper/types';
|
|||||||
|
|
||||||
export const useSelectSystem = () => {
|
export const useSelectSystem = () => {
|
||||||
const rf = useReactFlow();
|
const rf = useReactFlow();
|
||||||
|
|
||||||
const ref = useRef({ rf });
|
const ref = useRef({ rf });
|
||||||
ref.current = { rf };
|
ref.current = { rf };
|
||||||
|
|
||||||
return useCallback((systemId: CommandSelectSystem) => {
|
return useCallback((systemId: CommandSelectSystem) => {
|
||||||
if (!ref.current?.rf) {
|
ref.current.rf.setNodes(nds =>
|
||||||
return;
|
nds.map(node => {
|
||||||
}
|
return {
|
||||||
const systemNode = ref.current.rf.getNodes().find(x => x.data.id === systemId);
|
...node,
|
||||||
if (!systemNode) {
|
selected: node.id === systemId,
|
||||||
return;
|
};
|
||||||
}
|
}),
|
||||||
|
);
|
||||||
ref.current.rf.setCenter(systemNode.position.x, systemNode.position.y, { duration: 1000 });
|
|
||||||
}, []);
|
}, []);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { ForwardedRef, useImperativeHandle } from 'react';
|
import { ForwardedRef, useImperativeHandle, useRef } from 'react';
|
||||||
import {
|
import {
|
||||||
CommandAddConnections,
|
CommandAddConnections,
|
||||||
CommandAddSystems,
|
CommandAddSystems,
|
||||||
@@ -18,6 +18,9 @@ import {
|
|||||||
CommandUpdateSystems,
|
CommandUpdateSystems,
|
||||||
MapHandlers,
|
MapHandlers,
|
||||||
} from '@/hooks/Mapper/types/mapHandlers.ts';
|
} from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
|
||||||
|
import { useMapEventListener } from '@/hooks/Mapper/events';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
useCommandsCharacters,
|
useCommandsCharacters,
|
||||||
useCommandsConnections,
|
useCommandsConnections,
|
||||||
@@ -26,6 +29,7 @@ import {
|
|||||||
useMapInit,
|
useMapInit,
|
||||||
useMapRemoveSystems,
|
useMapRemoveSystems,
|
||||||
useMapUpdateSystems,
|
useMapUpdateSystems,
|
||||||
|
useCenterSystem,
|
||||||
useSelectSystem,
|
useSelectSystem,
|
||||||
} from './api';
|
} from './api';
|
||||||
import { OnMapSelectionChange } from '@/hooks/Mapper/components/map/map.types.ts';
|
import { OnMapSelectionChange } from '@/hooks/Mapper/components/map/map.types.ts';
|
||||||
@@ -35,8 +39,12 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
|
|||||||
const mapAddSystems = useMapAddSystems();
|
const mapAddSystems = useMapAddSystems();
|
||||||
const mapUpdateSystems = useMapUpdateSystems();
|
const mapUpdateSystems = useMapUpdateSystems();
|
||||||
const removeSystems = useMapRemoveSystems(onSelectionChange);
|
const removeSystems = useMapRemoveSystems(onSelectionChange);
|
||||||
|
const centerSystem = useCenterSystem();
|
||||||
const selectSystem = useSelectSystem();
|
const selectSystem = useSelectSystem();
|
||||||
|
|
||||||
|
const selectRef = useRef({ onSelectionChange });
|
||||||
|
selectRef.current = { onSelectionChange };
|
||||||
|
|
||||||
const { addConnections, removeConnections, updateConnection } = useCommandsConnections();
|
const { addConnections, removeConnections, updateConnection } = useCommandsConnections();
|
||||||
const { mapUpdated, killsUpdated } = useMapCommands();
|
const { mapUpdated, killsUpdated } = useMapCommands();
|
||||||
const { charactersUpdated, presentCharacters, characterAdded, characterRemoved, characterUpdated } =
|
const { charactersUpdated, presentCharacters, characterAdded, characterRemoved, characterUpdated } =
|
||||||
@@ -52,16 +60,13 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
|
|||||||
mapInit(data as CommandInit);
|
mapInit(data as CommandInit);
|
||||||
break;
|
break;
|
||||||
case Commands.addSystems:
|
case Commands.addSystems:
|
||||||
mapAddSystems(data as CommandAddSystems);
|
|
||||||
break;
|
break;
|
||||||
case Commands.updateSystems:
|
case Commands.updateSystems:
|
||||||
mapUpdateSystems(data as CommandUpdateSystems);
|
mapUpdateSystems(data as CommandUpdateSystems);
|
||||||
break;
|
break;
|
||||||
case Commands.removeSystems:
|
case Commands.removeSystems:
|
||||||
removeSystems(data as CommandRemoveSystems);
|
|
||||||
break;
|
break;
|
||||||
case Commands.addConnections:
|
case Commands.addConnections:
|
||||||
addConnections(data as CommandAddConnections);
|
|
||||||
break;
|
break;
|
||||||
case Commands.removeConnections:
|
case Commands.removeConnections:
|
||||||
removeConnections(data as CommandRemoveConnections);
|
removeConnections(data as CommandRemoveConnections);
|
||||||
@@ -91,14 +96,32 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
|
|||||||
killsUpdated(data as CommandKillsUpdated);
|
killsUpdated(data as CommandKillsUpdated);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case Commands.centerSystem:
|
||||||
|
setTimeout(() => {
|
||||||
|
const systemId = `${data}`;
|
||||||
|
centerSystem(systemId as CommandSelectSystem);
|
||||||
|
}, 100);
|
||||||
|
break;
|
||||||
|
|
||||||
case Commands.selectSystem:
|
case Commands.selectSystem:
|
||||||
selectSystem(data as CommandSelectSystem);
|
setTimeout(() => {
|
||||||
|
const systemId = `${data}`;
|
||||||
|
selectRef.current.onSelectionChange({
|
||||||
|
systems: [systemId],
|
||||||
|
connections: [],
|
||||||
|
});
|
||||||
|
selectSystem(systemId as CommandSelectSystem);
|
||||||
|
}, 100);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case Commands.routes:
|
case Commands.routes:
|
||||||
// do nothing here
|
// do nothing here
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case Commands.linkSignatureToSystem:
|
||||||
|
// do nothing here
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
console.warn(`Map handlers: Unknown command: ${type}`, data);
|
console.warn(`Map handlers: Unknown command: ${type}`, data);
|
||||||
break;
|
break;
|
||||||
@@ -108,4 +131,20 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
|
|||||||
},
|
},
|
||||||
[],
|
[],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
useMapEventListener(event => {
|
||||||
|
switch (event.name) {
|
||||||
|
case Commands.addConnections:
|
||||||
|
addConnections(event.data as CommandAddConnections);
|
||||||
|
break;
|
||||||
|
case Commands.addSystems:
|
||||||
|
mapAddSystems(event.data as CommandAddSystems);
|
||||||
|
break;
|
||||||
|
case Commands.removeSystems:
|
||||||
|
removeSystems(event.data as CommandRemoveSystems);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import { Button } from 'primereact/button';
|
|||||||
import { OutCommand } from '@/hooks/Mapper/types';
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
import { IconField } from 'primereact/iconfield';
|
import { IconField } from 'primereact/iconfield';
|
||||||
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager.ts';
|
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager.ts';
|
||||||
import { WdImageSize, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
import { WdImageSize, WdImgButton, TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
interface SystemCustomLabelDialog {
|
interface SystemCustomLabelDialog {
|
||||||
systemId: string;
|
systemId: string;
|
||||||
@@ -79,14 +79,14 @@ export const SystemCustomLabelDialog = ({ systemId, visible, setVisible }: Syste
|
|||||||
|
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
const handleInput = useCallback(e => {
|
const handleInput = useCallback(e => {
|
||||||
e.target.value = e.target.value.toUpperCase().replace(/[^A-Z0-9]/g, '');
|
e.target.value = e.target.value.toUpperCase().replace(/[^A-Z0-9\-[\](){}]/g, '');
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
header="Edit label"
|
header="Edit label"
|
||||||
visible={visible}
|
visible={visible}
|
||||||
draggable={false}
|
draggable={true}
|
||||||
style={{ width: '250px' }}
|
style={{ width: '250px' }}
|
||||||
onHide={onHide}
|
onHide={onHide}
|
||||||
onShow={onShow}
|
onShow={onShow}
|
||||||
@@ -100,9 +100,13 @@ export const SystemCustomLabelDialog = ({ systemId, visible, setVisible }: Syste
|
|||||||
<IconField>
|
<IconField>
|
||||||
{label !== '' && (
|
{label !== '' && (
|
||||||
<WdImgButton
|
<WdImgButton
|
||||||
className="pi pi-trash p-input-icon"
|
className="pi pi-trash text-red-400"
|
||||||
textSize={WdImageSize.large}
|
textSize={WdImageSize.large}
|
||||||
tooltip={{ content: 'Reset label' }}
|
tooltip={{
|
||||||
|
content: 'Remove custom label',
|
||||||
|
className: 'pi p-input-icon',
|
||||||
|
position: TooltipPosition.top,
|
||||||
|
}}
|
||||||
onClick={handleReset}
|
onClick={handleReset}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
@@ -111,7 +115,7 @@ export const SystemCustomLabelDialog = ({ systemId, visible, setVisible }: Syste
|
|||||||
aria-describedby="username-help"
|
aria-describedby="username-help"
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
value={label}
|
value={label}
|
||||||
maxLength={3}
|
maxLength={5}
|
||||||
onChange={e => setLabel(e.target.value)}
|
onChange={e => setLabel(e.target.value)}
|
||||||
// @ts-expect-error
|
// @ts-expect-error
|
||||||
ref={inputRef}
|
ref={inputRef}
|
||||||
|
|||||||
@@ -0,0 +1,68 @@
|
|||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
import { Dialog } from 'primereact/dialog';
|
||||||
|
|
||||||
|
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 {
|
||||||
|
Setting,
|
||||||
|
COSMIC_SIGNATURE,
|
||||||
|
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SystemSignatureSettingsDialog';
|
||||||
|
|
||||||
|
interface SystemLinkSignatureDialogProps {
|
||||||
|
data: CommandLinkSignatureToSystem;
|
||||||
|
setVisible: (visible: boolean) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const signatureSettings: Setting[] = [{ key: COSMIC_SIGNATURE, name: 'Show Cosmic Signatures', value: true }];
|
||||||
|
|
||||||
|
export const SystemLinkSignatureDialog = ({ data, setVisible }: SystemLinkSignatureDialogProps) => {
|
||||||
|
const { outCommand } = useMapRootState();
|
||||||
|
|
||||||
|
const ref = useRef({ outCommand });
|
||||||
|
ref.current = { outCommand };
|
||||||
|
|
||||||
|
const handleHide = useCallback(() => {
|
||||||
|
setVisible(false);
|
||||||
|
}, [setVisible]);
|
||||||
|
|
||||||
|
const handleSelect = useCallback(
|
||||||
|
(signature: SystemSignature) => {
|
||||||
|
if (!signature) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { outCommand } = ref.current;
|
||||||
|
|
||||||
|
outCommand({
|
||||||
|
type: OutCommand.linkSignatureToSystem,
|
||||||
|
data: {
|
||||||
|
...data,
|
||||||
|
signature_eve_id: signature.eve_id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
setVisible(false);
|
||||||
|
},
|
||||||
|
[data, setVisible],
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog
|
||||||
|
header="Select signature to link"
|
||||||
|
visible
|
||||||
|
draggable={false}
|
||||||
|
style={{ width: '500px' }}
|
||||||
|
onHide={handleHide}
|
||||||
|
contentClassName="!p-0"
|
||||||
|
>
|
||||||
|
<SystemSignaturesContent
|
||||||
|
systemId={`${data.solar_system_source}`}
|
||||||
|
settings={signatureSettings}
|
||||||
|
onSelect={handleSelect}
|
||||||
|
selectable={true}
|
||||||
|
/>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SystemLinkSignatureDialog';
|
||||||
@@ -90,7 +90,7 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleInput = useCallback((e: any) => {
|
const handleInput = useCallback((e: any) => {
|
||||||
e.target.value = e.target.value.toUpperCase().replace(/[^A-Z0-9]/g, '');
|
e.target.value = e.target.value.toUpperCase().replace(/[^A-Z0-9\-[\](){}]/g, '');
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -160,7 +160,7 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
|
|||||||
aria-describedby="label"
|
aria-describedby="label"
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
value={label}
|
value={label}
|
||||||
maxLength={3}
|
maxLength={5}
|
||||||
onChange={e => setLabel(e.target.value)}
|
onChange={e => setLabel(e.target.value)}
|
||||||
onInput={handleInput}
|
onInput={handleInput}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -2,3 +2,4 @@ export * from './Widget';
|
|||||||
export * from './WidgetsGrid';
|
export * from './WidgetsGrid';
|
||||||
export * from './SystemSettingsDialog';
|
export * from './SystemSettingsDialog';
|
||||||
export * from './SystemCustomLabelDialog';
|
export * from './SystemCustomLabelDialog';
|
||||||
|
export * from './SystemLinkSignatureDialog';
|
||||||
|
|||||||
@@ -91,7 +91,7 @@ export const RoutesList = ({ data, onContextMenu }: RoutesListProps) => {
|
|||||||
const { mapRef } = useMapRootState();
|
const { mapRef } = useMapRootState();
|
||||||
|
|
||||||
const handleClick = useCallback(
|
const handleClick = useCallback(
|
||||||
(systemId: number) => mapRef.current?.command(Commands.selectSystem, systemId.toString()),
|
(systemId: number) => mapRef.current?.command(Commands.centerSystem, systemId.toString()),
|
||||||
[mapRef],
|
[mapRef],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ export const RoutesWidgetContent = () => {
|
|||||||
|
|
||||||
const { loading } = useLoadRoutes();
|
const { loading } = useLoadRoutes();
|
||||||
|
|
||||||
const { systems: systemStatics, loadSystems } = useLoadSystemStatic({ systems: hubs ?? [] });
|
const { systems: systemStatics, loadSystems, lastUpdateKey } = useLoadSystemStatic({ systems: hubs ?? [] });
|
||||||
const { open, ...systemCtxProps } = useContextMenuSystemInfoHandlers({
|
const { open, ...systemCtxProps } = useContextMenuSystemInfoHandlers({
|
||||||
outCommand,
|
outCommand,
|
||||||
hubs,
|
hubs,
|
||||||
@@ -51,9 +51,10 @@ export const RoutesWidgetContent = () => {
|
|||||||
|
|
||||||
return { ...systemStatics.get(parseInt(x))!, ...(sys && { customName: sys.name ?? '' }) };
|
return { ...systemStatics.get(parseInt(x))!, ...(sys && { customName: sys.name ?? '' }) };
|
||||||
});
|
});
|
||||||
}, [hubs, systems, systemStatics]);
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [hubs, systems, systemStatics, lastUpdateKey]);
|
||||||
|
|
||||||
const preparedRoutes = useMemo(() => {
|
const preparedRoutes: Route[] = useMemo(() => {
|
||||||
return (
|
return (
|
||||||
routes?.routes
|
routes?.routes
|
||||||
.sort(sortByDist)
|
.sort(sortByDist)
|
||||||
@@ -70,15 +71,17 @@ export const RoutesWidgetContent = () => {
|
|||||||
);
|
);
|
||||||
}, [routes?.routes, routes?.systems_static_data, systemId]);
|
}, [routes?.routes, routes?.systems_static_data, systemId]);
|
||||||
|
|
||||||
const refData = useRef({ open, loadSystems });
|
const refData = useRef({ open, loadSystems, preparedRoutes });
|
||||||
refData.current = { open, loadSystems };
|
refData.current = { open, loadSystems, preparedRoutes };
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
(async () => await refData.current.loadSystems(hubs))();
|
(async () => await refData.current.loadSystems(hubs))();
|
||||||
}, [hubs]);
|
}, [hubs]);
|
||||||
|
|
||||||
const handleClick = useCallback((e: MouseEvent, systemId: string) => {
|
const handleClick = useCallback((e: MouseEvent, systemId: string) => {
|
||||||
refData.current.open(e, systemId);
|
const route = refData.current.preparedRoutes.find(x => x.destination.toString() === systemId);
|
||||||
|
|
||||||
|
refData.current.open(e, systemId, route?.mapped_systems ?? []);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleContextMenu = useCallback(
|
const handleContextMenu = useCallback(
|
||||||
@@ -114,6 +117,10 @@ export const RoutesWidgetContent = () => {
|
|||||||
{preparedRoutes.map(route => {
|
{preparedRoutes.map(route => {
|
||||||
const sys = preparedHubs.find(x => x.solar_system_id === route.destination)!;
|
const sys = preparedHubs.find(x => x.solar_system_id === route.destination)!;
|
||||||
|
|
||||||
|
// TODO do not delte this console log
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
// console.log('JOipP', `Check sys [${route.destination}]:`, sys);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="flex gap-2 items-center">
|
<div className="flex gap-2 items-center">
|
||||||
@@ -141,7 +148,14 @@ export const RoutesWidgetContent = () => {
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<ContextMenuSystemInfo hubs={hubs} systems={systems} systemStatics={systemStatics} {...systemCtxProps} />
|
<ContextMenuSystemInfo
|
||||||
|
hubs={hubs}
|
||||||
|
routes={preparedRoutes}
|
||||||
|
systems={systems}
|
||||||
|
systemStatics={systemStatics}
|
||||||
|
systemIdFrom={systemId}
|
||||||
|
{...systemCtxProps}
|
||||||
|
/>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -13,9 +13,10 @@ export const SystemInfoContent = ({ systemId }: SystemInfoContentProps) => {
|
|||||||
data: { systems, wormholesData },
|
data: { systems, wormholesData },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const sys = getSystemById(systems, systemId)!;
|
const sys = getSystemById(systems, systemId)! || {};
|
||||||
const { description } = sys;
|
const { description } = sys;
|
||||||
const { system_class, region_name, constellation_name, statics, effect_name, effect_power } = sys.system_static_info;
|
const { system_class, region_name, constellation_name, statics, effect_name, effect_power } =
|
||||||
|
sys.system_static_info || {};
|
||||||
const isWH = isWormholeSpace(system_class);
|
const isWH = isWormholeSpace(system_class);
|
||||||
const sortedStatics = useMemo(() => sortWHClasses(wormholesData, statics), [wormholesData, statics]);
|
const sortedStatics = useMemo(() => sortWHClasses(wormholesData, statics), [wormholesData, statics]);
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,14 @@ import { Checkbox } from 'primereact/checkbox';
|
|||||||
|
|
||||||
export type Setting = { key: string; name: string; value: boolean };
|
export type Setting = { key: string; name: string; value: boolean };
|
||||||
|
|
||||||
|
export const COSMIC_SIGNATURE = 'Cosmic Signature';
|
||||||
|
export const COSMIC_ANOMALY = 'Cosmic Anomaly';
|
||||||
|
export const DEPLOYABLE = 'Deployable';
|
||||||
|
export const STRUCTURE = 'Structure';
|
||||||
|
export const STARBASE = 'Starbase';
|
||||||
|
export const SHIP = 'Ship';
|
||||||
|
export const DRONE = 'Drone';
|
||||||
|
|
||||||
interface SystemSignatureSettingsDialogProps {
|
interface SystemSignatureSettingsDialogProps {
|
||||||
settings: Setting[];
|
settings: Setting[];
|
||||||
onSave: (settings: Setting[]) => void;
|
onSave: (settings: Setting[]) => void;
|
||||||
|
|||||||
@@ -1,7 +1,17 @@
|
|||||||
import { Widget } from '@/hooks/Mapper/components/mapInterface/components';
|
import { Widget } from '@/hooks/Mapper/components/mapInterface/components';
|
||||||
import { InfoDrawer, LayoutEventBlocker, TooltipPosition, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
import { InfoDrawer, LayoutEventBlocker, TooltipPosition, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
||||||
import { SystemSignaturesContent } from './SystemSignaturesContent';
|
import { SystemSignaturesContent } from './SystemSignaturesContent';
|
||||||
import { Setting, SystemSignatureSettingsDialog } from './SystemSignatureSettingsDialog';
|
import {
|
||||||
|
Setting,
|
||||||
|
SystemSignatureSettingsDialog,
|
||||||
|
COSMIC_SIGNATURE,
|
||||||
|
COSMIC_ANOMALY,
|
||||||
|
DEPLOYABLE,
|
||||||
|
STRUCTURE,
|
||||||
|
STARBASE,
|
||||||
|
SHIP,
|
||||||
|
DRONE,
|
||||||
|
} from './SystemSignatureSettingsDialog';
|
||||||
|
|
||||||
import React, { useCallback, useEffect, useState } from 'react';
|
import React, { useCallback, useEffect, useState } from 'react';
|
||||||
|
|
||||||
@@ -9,14 +19,6 @@ import { PrimeIcons } from 'primereact/api';
|
|||||||
|
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
export const COSMIC_SIGNATURE = 'Cosmic Signature';
|
|
||||||
export const COSMIC_ANOMALY = 'Cosmic Anomaly';
|
|
||||||
export const DEPLOYABLE = 'Deployable';
|
|
||||||
export const STRUCTURE = 'Structure';
|
|
||||||
export const STARBASE = 'Starbase';
|
|
||||||
export const SHIP = 'Ship';
|
|
||||||
export const DRONE = 'Drone';
|
|
||||||
|
|
||||||
const settings: Setting[] = [
|
const settings: Setting[] = [
|
||||||
{ key: COSMIC_ANOMALY, name: 'Show Anomalies', value: true },
|
{ key: COSMIC_ANOMALY, name: 'Show Anomalies', value: true },
|
||||||
{ key: COSMIC_SIGNATURE, name: 'Show Cosmic Signatures', value: true },
|
{ key: COSMIC_SIGNATURE, name: 'Show Cosmic Signatures', value: true },
|
||||||
|
|||||||
@@ -4,3 +4,7 @@
|
|||||||
font-size: 12px !important;
|
font-size: 12px !important;
|
||||||
line-height: 8px;
|
line-height: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.Table {
|
||||||
|
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { useClipboard } from '@/hooks/Mapper/hooks/useClipboard';
|
import { useClipboard } from '@/hooks/Mapper/hooks/useClipboard';
|
||||||
import { parseSignatures } from '@/hooks/Mapper/helpers';
|
import { parseSignatures } from '@/hooks/Mapper/helpers';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { Commands, OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { WdTooltip, WdTooltipHandlers } from '@/hooks/Mapper/components/ui-kit';
|
import { WdTooltip, WdTooltipHandlers } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
import { DataTable, DataTableRowMouseEvent } from 'primereact/datatable';
|
import { DataTable, DataTableRowClickEvent, DataTableRowMouseEvent, SortOrder } from 'primereact/datatable';
|
||||||
import { Column } from 'primereact/column';
|
import { Column } from 'primereact/column';
|
||||||
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||||
import useRefState from 'react-usestateref';
|
import useRefState from 'react-usestateref';
|
||||||
@@ -22,23 +22,47 @@ import {
|
|||||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/helpers';
|
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/helpers';
|
||||||
import {
|
import {
|
||||||
renderIcon,
|
renderIcon,
|
||||||
renderName,
|
renderInfoColumn,
|
||||||
renderTimeLeft,
|
renderTimeLeft,
|
||||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/renders';
|
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/renders';
|
||||||
|
import useLocalStorageState from 'use-local-storage-state';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { SignatureSettings } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings';
|
||||||
|
import { useMapEventListener } from '@/hooks/Mapper/events';
|
||||||
|
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
||||||
|
|
||||||
|
type SystemSignaturesSortSettings = {
|
||||||
|
sortField: string;
|
||||||
|
sortOrder: SortOrder;
|
||||||
|
};
|
||||||
|
|
||||||
|
const SORT_DEFAULT_VALUES: SystemSignaturesSortSettings = {
|
||||||
|
sortField: 'updated_at',
|
||||||
|
sortOrder: -1,
|
||||||
|
};
|
||||||
|
|
||||||
interface SystemSignaturesContentProps {
|
interface SystemSignaturesContentProps {
|
||||||
systemId: string;
|
systemId: string;
|
||||||
settings: Setting[];
|
settings: Setting[];
|
||||||
|
selectable?: boolean;
|
||||||
|
onSelect?: (signature: SystemSignature) => void;
|
||||||
}
|
}
|
||||||
export const SystemSignaturesContent = ({ systemId, settings }: SystemSignaturesContentProps) => {
|
export const SystemSignaturesContent = ({ systemId, settings, selectable, onSelect }: SystemSignaturesContentProps) => {
|
||||||
const { outCommand } = useMapRootState();
|
const { outCommand } = useMapRootState();
|
||||||
|
|
||||||
const [signatures, setSignatures, signaturesRef] = useRefState<SystemSignature[]>([]);
|
const [signatures, setSignatures, signaturesRef] = useRefState<SystemSignature[]>([]);
|
||||||
const [selectedSignatures, setSelectedSignatures] = useState<SystemSignature[]>([]);
|
const [selectedSignatures, setSelectedSignatures] = useState<SystemSignature[]>([]);
|
||||||
const [nameColumnWidth, setNameColumnWidth] = useState('auto');
|
const [nameColumnWidth, setNameColumnWidth] = useState('auto');
|
||||||
|
const [parsedSignatures, setParsedSignatures] = useState<SystemSignature[]>([]);
|
||||||
|
const [askUser, setAskUser] = useState(false);
|
||||||
|
const [selectedSignature, setSelectedSignature] = useState<SystemSignature | null>(null);
|
||||||
|
|
||||||
const [hoveredSig, setHoveredSig] = useState<SystemSignature | null>(null);
|
const [hoveredSig, setHoveredSig] = useState<SystemSignature | null>(null);
|
||||||
|
|
||||||
|
const [sortSettings, setSortSettings] = useLocalStorageState<SystemSignaturesSortSettings>('window:signatures:sort', {
|
||||||
|
defaultValue: SORT_DEFAULT_VALUES,
|
||||||
|
});
|
||||||
|
|
||||||
const tableRef = useRef<HTMLDivElement>(null);
|
const tableRef = useRef<HTMLDivElement>(null);
|
||||||
const compact = useMaxWidth(tableRef, 260);
|
const compact = useMaxWidth(tableRef, 260);
|
||||||
const medium = useMaxWidth(tableRef, 380);
|
const medium = useMaxWidth(tableRef, 380);
|
||||||
@@ -50,7 +74,7 @@ export const SystemSignaturesContent = ({ systemId, settings }: SystemSignatures
|
|||||||
const handleResize = useCallback(() => {
|
const handleResize = useCallback(() => {
|
||||||
if (tableRef.current) {
|
if (tableRef.current) {
|
||||||
const tableWidth = tableRef.current.offsetWidth;
|
const tableWidth = tableRef.current.offsetWidth;
|
||||||
const otherColumnsWidth = 265;
|
const otherColumnsWidth = 276;
|
||||||
const availableWidth = tableWidth - otherColumnsWidth;
|
const availableWidth = tableWidth - otherColumnsWidth;
|
||||||
setNameColumnWidth(`${availableWidth}px`);
|
setNameColumnWidth(`${availableWidth}px`);
|
||||||
}
|
}
|
||||||
@@ -70,12 +94,33 @@ export const SystemSignaturesContent = ({ systemId, settings }: SystemSignatures
|
|||||||
data: { system_id: systemId },
|
data: { system_id: systemId },
|
||||||
});
|
});
|
||||||
|
|
||||||
|
setAskUser(false);
|
||||||
setSignatures(signatures);
|
setSignatures(signatures);
|
||||||
}, [outCommand, systemId]);
|
}, [outCommand, systemId]);
|
||||||
|
|
||||||
|
// const updateSignatures = useCallback(
|
||||||
|
// async (newSignatures: SystemSignature[], updateOnly: boolean) => {
|
||||||
|
// const { added, updated, removed } = getActualSigs(signaturesRef.current, newSignatures, updateOnly);
|
||||||
|
|
||||||
|
// const { signatures: updatedSignatures } = await outCommand({
|
||||||
|
// type: OutCommand.updateSignatures,
|
||||||
|
// data: {
|
||||||
|
// system_id: systemId,
|
||||||
|
// added,
|
||||||
|
// updated,
|
||||||
|
// removed,
|
||||||
|
// },
|
||||||
|
// });
|
||||||
|
|
||||||
|
// setSignatures(() => updatedSignatures);
|
||||||
|
// setSelectedSignatures([]);
|
||||||
|
// },
|
||||||
|
// [outCommand, systemId],
|
||||||
|
// );
|
||||||
|
|
||||||
const handleUpdateSignatures = useCallback(
|
const handleUpdateSignatures = useCallback(
|
||||||
async (newSignatures: SystemSignature[]) => {
|
async (newSignatures: SystemSignature[], updateOnly: boolean) => {
|
||||||
const { added, updated, removed } = getActualSigs(signaturesRef.current, newSignatures);
|
const { added, updated, removed } = getActualSigs(signaturesRef.current, newSignatures, updateOnly);
|
||||||
|
|
||||||
const { signatures: updatedSignatures } = await outCommand({
|
const { signatures: updatedSignatures } = await outCommand({
|
||||||
type: OutCommand.updateSignatures,
|
type: OutCommand.updateSignatures,
|
||||||
@@ -94,43 +139,96 @@ export const SystemSignaturesContent = ({ systemId, settings }: SystemSignatures
|
|||||||
);
|
);
|
||||||
|
|
||||||
const handleDeleteSelected = useCallback(async () => {
|
const handleDeleteSelected = useCallback(async () => {
|
||||||
|
if (selectable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
if (selectedSignatures.length === 0) {
|
if (selectedSignatures.length === 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const selectedSignaturesEveIds = selectedSignatures.map(x => x.eve_id);
|
const selectedSignaturesEveIds = selectedSignatures.map(x => x.eve_id);
|
||||||
await handleUpdateSignatures(signatures.filter(x => !selectedSignaturesEveIds.includes(x.eve_id)));
|
await handleUpdateSignatures(
|
||||||
}, [handleUpdateSignatures, signatures, selectedSignatures]);
|
signatures.filter(x => !selectedSignaturesEveIds.includes(x.eve_id)),
|
||||||
|
false,
|
||||||
|
);
|
||||||
|
}, [handleUpdateSignatures, selectable, signatures, selectedSignatures]);
|
||||||
|
|
||||||
const handleSelectAll = useCallback(() => {
|
const handleSelectAll = useCallback(() => {
|
||||||
setSelectedSignatures(signatures);
|
setSelectedSignatures(signatures);
|
||||||
}, [signatures]);
|
}, [signatures]);
|
||||||
|
|
||||||
|
const handleReplaceAll = useCallback(() => {
|
||||||
|
handleUpdateSignatures(parsedSignatures, false);
|
||||||
|
setAskUser(false);
|
||||||
|
}, [parsedSignatures, handleUpdateSignatures]);
|
||||||
|
|
||||||
|
const handleUpdateOnly = useCallback(() => {
|
||||||
|
handleUpdateSignatures(parsedSignatures, true);
|
||||||
|
setAskUser(false);
|
||||||
|
}, [parsedSignatures, handleUpdateSignatures]);
|
||||||
|
|
||||||
|
const handleSelectSignatures = useCallback(
|
||||||
|
// TODO still will be good to define types if we use typescript
|
||||||
|
// @ts-ignore
|
||||||
|
e => {
|
||||||
|
if (selectable) {
|
||||||
|
onSelect?.(e.value);
|
||||||
|
} else {
|
||||||
|
setSelectedSignatures(e.value);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[onSelect, selectable],
|
||||||
|
);
|
||||||
|
|
||||||
useHotkey(true, ['a'], handleSelectAll);
|
useHotkey(true, ['a'], handleSelectAll);
|
||||||
|
|
||||||
useHotkey(false, ['Backspace', 'Delete'], handleDeleteSelected);
|
useHotkey(false, ['Backspace', 'Delete'], handleDeleteSelected);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
if (selectable) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!clipboardContent) {
|
if (!clipboardContent) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const signatures = parseSignatures(
|
const newSignatures = parseSignatures(
|
||||||
clipboardContent,
|
clipboardContent,
|
||||||
settings.map(x => x.key),
|
settings.map(x => x.key),
|
||||||
);
|
);
|
||||||
|
|
||||||
handleUpdateSignatures(signatures);
|
const { removed } = getActualSigs(signaturesRef.current, newSignatures, false);
|
||||||
}, [clipboardContent]);
|
|
||||||
|
if (!signaturesRef.current || !signaturesRef.current.length || !removed.length) {
|
||||||
|
handleUpdateSignatures(newSignatures, false);
|
||||||
|
} else {
|
||||||
|
setParsedSignatures(newSignatures);
|
||||||
|
setAskUser(true);
|
||||||
|
}
|
||||||
|
}, [clipboardContent, selectable]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!systemId) {
|
if (!systemId) {
|
||||||
setSignatures([]);
|
setSignatures([]);
|
||||||
|
setAskUser(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
handleGetSignatures();
|
handleGetSignatures();
|
||||||
}, [systemId]);
|
}, [systemId]);
|
||||||
|
|
||||||
|
useMapEventListener(event => {
|
||||||
|
switch (event.name) {
|
||||||
|
case Commands.signaturesUpdated:
|
||||||
|
if (event.data?.toString() !== systemId.toString()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
handleGetSignatures();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const observer = new ResizeObserver(handleResize);
|
const observer = new ResizeObserver(handleResize);
|
||||||
if (tableRef.current) {
|
if (tableRef.current) {
|
||||||
@@ -159,83 +257,147 @@ export const SystemSignaturesContent = ({ systemId, settings }: SystemSignatures
|
|||||||
setHoveredSig(null);
|
setHoveredSig(null);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const renderToolbar = (/*row: SystemSignature*/) => {
|
||||||
|
return (
|
||||||
|
<div className="flex justify-end items-center gap-2 mr-[4px]">
|
||||||
|
<WdTooltipWrapper content="To Edit Signature do double click">
|
||||||
|
<span className={clsx(PrimeIcons.PENCIL, 'text-[10px]')}></span>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const [showSignatureSettings, setShowSignatureSettings] = useState(false);
|
||||||
|
|
||||||
|
const handleRowClick = (e: DataTableRowClickEvent) => {
|
||||||
|
setSelectedSignature(e.data as SystemSignature);
|
||||||
|
setShowSignatureSettings(true);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div ref={tableRef} className="h-full">
|
<>
|
||||||
{filteredSignatures.length === 0 ? (
|
<div ref={tableRef} className={'h-full '}>
|
||||||
<div className="w-full h-full flex justify-center items-center select-none text-stone-400/80 text-sm">
|
{filteredSignatures.length === 0 ? (
|
||||||
No signatures
|
<div className="w-full h-full flex justify-center items-center select-none text-stone-400/80 text-sm">
|
||||||
</div>
|
No signatures
|
||||||
) : (
|
</div>
|
||||||
<>
|
) : (
|
||||||
<DataTable
|
<>
|
||||||
value={filteredSignatures}
|
{/* @ts-ignore */}
|
||||||
size="small"
|
<DataTable
|
||||||
selectionMode="multiple"
|
className={classes.Table}
|
||||||
selection={selectedSignatures}
|
value={filteredSignatures}
|
||||||
onSelectionChange={e => setSelectedSignatures(e.value)}
|
size="small"
|
||||||
dataKey="eve_id"
|
selectionMode={selectable ? 'single' : 'multiple'}
|
||||||
tableClassName="w-full select-none"
|
selection={selectedSignatures}
|
||||||
resizableColumns
|
metaKeySelection
|
||||||
rowHover
|
onSelectionChange={handleSelectSignatures}
|
||||||
selectAll
|
dataKey="eve_id"
|
||||||
showHeaders={false}
|
tableClassName="w-full select-none"
|
||||||
onRowMouseEnter={handleEnterRow}
|
resizableColumns={false}
|
||||||
onRowMouseLeave={handleLeaveRow}
|
onRowDoubleClick={handleRowClick}
|
||||||
rowClassName={row => {
|
rowHover
|
||||||
if (selectedSignatures.some(x => x.eve_id === row.eve_id)) {
|
selectAll
|
||||||
return clsx(classes.TableRowCompact, 'bg-amber-500/50 hover:bg-amber-500/70 transition duration-200');
|
sortField={sortSettings.sortField}
|
||||||
}
|
sortOrder={sortSettings.sortOrder}
|
||||||
|
onSort={event => setSortSettings(() => ({ sortField: event.sortField, sortOrder: event.sortOrder }))}
|
||||||
|
onRowMouseEnter={compact || medium ? handleEnterRow : undefined}
|
||||||
|
onRowMouseLeave={compact || medium ? handleLeaveRow : undefined}
|
||||||
|
rowClassName={row => {
|
||||||
|
if (selectedSignatures.some(x => x.eve_id === row.eve_id)) {
|
||||||
|
return clsx(classes.TableRowCompact, 'bg-amber-500/50 hover:bg-amber-500/70 transition duration-200');
|
||||||
|
}
|
||||||
|
|
||||||
const dateClass = getRowColorByTimeLeft(row.updated_at ? new Date(row.updated_at) : undefined);
|
const dateClass = getRowColorByTimeLeft(row.updated_at ? new Date(row.updated_at) : undefined);
|
||||||
if (!dateClass) {
|
if (!dateClass) {
|
||||||
return clsx(classes.TableRowCompact, 'hover:bg-purple-400/20 transition duration-200');
|
return clsx(classes.TableRowCompact, 'hover:bg-purple-400/20 transition duration-200');
|
||||||
}
|
}
|
||||||
|
|
||||||
return clsx(classes.TableRowCompact, dateClass);
|
return clsx(classes.TableRowCompact, dateClass);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Column
|
<Column
|
||||||
bodyClassName="p-0 px-1"
|
bodyClassName="p-0 px-1"
|
||||||
field="group"
|
field="group"
|
||||||
body={renderIcon}
|
body={x => renderIcon(x)}
|
||||||
style={{ maxWidth: 26, minWidth: 26, width: 26 }}
|
style={{ maxWidth: 26, minWidth: 26, width: 26, height: 25 }}
|
||||||
></Column>
|
></Column>
|
||||||
|
|
||||||
<Column
|
<Column
|
||||||
field="eve_id"
|
field="eve_id"
|
||||||
header="Id"
|
header="Id"
|
||||||
bodyClassName="text-ellipsis overflow-hidden whitespace-nowrap"
|
bodyClassName="text-ellipsis overflow-hidden whitespace-nowrap"
|
||||||
style={{ maxWidth: 72, minWidth: 72, width: 72 }}
|
style={{ maxWidth: 72, minWidth: 72, width: 72 }}
|
||||||
></Column>
|
sortable
|
||||||
<Column
|
></Column>
|
||||||
field="group"
|
<Column
|
||||||
header="Group"
|
field="group"
|
||||||
bodyClassName="text-ellipsis overflow-hidden whitespace-nowrap"
|
header="Group"
|
||||||
hidden={compact}
|
bodyClassName="text-ellipsis overflow-hidden whitespace-nowrap"
|
||||||
></Column>
|
hidden={compact}
|
||||||
<Column
|
sortable
|
||||||
field="name"
|
></Column>
|
||||||
header="Name"
|
<Column
|
||||||
bodyClassName="text-ellipsis overflow-hidden whitespace-nowrap"
|
field="info"
|
||||||
body={renderName}
|
// header="Info"
|
||||||
style={{ maxWidth: nameColumnWidth }}
|
bodyClassName="text-ellipsis overflow-hidden whitespace-nowrap"
|
||||||
hidden={compact || medium}
|
body={renderInfoColumn}
|
||||||
></Column>
|
style={{ maxWidth: nameColumnWidth }}
|
||||||
<Column
|
hidden={compact || medium}
|
||||||
field="updated_at"
|
></Column>
|
||||||
header="Updated"
|
<Column
|
||||||
dataType="date"
|
field="updated_at"
|
||||||
bodyClassName="w-[80px] text-ellipsis overflow-hidden whitespace-nowrap"
|
header="Updated"
|
||||||
body={renderTimeLeft}
|
dataType="date"
|
||||||
></Column>
|
bodyClassName="w-[70px] text-ellipsis overflow-hidden whitespace-nowrap"
|
||||||
</DataTable>
|
body={renderTimeLeft}
|
||||||
</>
|
sortable
|
||||||
)}
|
></Column>
|
||||||
<WdTooltip
|
|
||||||
className="bg-stone-900/95 text-slate-50"
|
<Column
|
||||||
ref={tooltipRef}
|
bodyClassName="p-0 pl-1 pr-2"
|
||||||
content={hoveredSig ? <SignatureView {...hoveredSig} /> : null}
|
field="group"
|
||||||
/>
|
body={renderToolbar}
|
||||||
</div>
|
// headerClassName={headerClasses}
|
||||||
|
style={{ maxWidth: 26, minWidth: 26, width: 26 }}
|
||||||
|
></Column>
|
||||||
|
</DataTable>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
<WdTooltip
|
||||||
|
className="bg-stone-900/95 text-slate-50"
|
||||||
|
ref={tooltipRef}
|
||||||
|
content={hoveredSig ? <SignatureView {...hoveredSig} /> : null}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<SignatureSettings
|
||||||
|
systemId={systemId}
|
||||||
|
show={showSignatureSettings}
|
||||||
|
onHide={() => setShowSignatureSettings(false)}
|
||||||
|
signatureData={selectedSignature}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{askUser && (
|
||||||
|
<div className="absolute left-[1px] top-[29px] h-[calc(100%-30px)] w-[calc(100%-3px)] bg-stone-900/10 backdrop-blur-sm">
|
||||||
|
<div className="absolute top-0 left-0 w-full h-full flex flex-col items-center justify-center">
|
||||||
|
<div className="text-stone-400/80 text-sm">
|
||||||
|
<div className="flex flex-col text-center gap-2">
|
||||||
|
<button className="p-button p-component p-button-outlined p-button-sm btn-wide">
|
||||||
|
<span className="p-button-label p-c" onClick={handleUpdateOnly}>
|
||||||
|
Update
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
<button className="p-button p-component p-button-outlined p-button-sm btn-wide">
|
||||||
|
<span className="p-button-label p-c" onClick={handleReplaceAll}>
|
||||||
|
Update & Delete missing
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { getState } from './getState.ts';
|
|||||||
export const getActualSigs = (
|
export const getActualSigs = (
|
||||||
oldSignatures: SystemSignature[],
|
oldSignatures: SystemSignature[],
|
||||||
newSignatures: SystemSignature[],
|
newSignatures: SystemSignature[],
|
||||||
|
updateOnly: boolean,
|
||||||
): { added: SystemSignature[]; updated: SystemSignature[]; removed: SystemSignature[] } => {
|
): { added: SystemSignature[]; updated: SystemSignature[]; removed: SystemSignature[] } => {
|
||||||
const updated: SystemSignature[] = [];
|
const updated: SystemSignature[] = [];
|
||||||
const removed: SystemSignature[] = [];
|
const removed: SystemSignature[] = [];
|
||||||
@@ -20,7 +21,9 @@ export const getActualSigs = (
|
|||||||
updated.push({ ...oldSig, group: newSig.group, name: newSig.name });
|
updated.push({ ...oldSig, group: newSig.group, name: newSig.name });
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
removed.push(oldSig);
|
if (!updateOnly) {
|
||||||
|
removed.push(oldSig);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -7,9 +7,9 @@ export const getState = (_: string[], newSig: SystemSignature) => {
|
|||||||
let state = -1;
|
let state = -1;
|
||||||
if (!newSig.group || newSig.group === '') {
|
if (!newSig.group || newSig.group === '') {
|
||||||
state = 0;
|
state = 0;
|
||||||
} else if (!!newSig.group && newSig.group !== '' && newSig.name === '') {
|
} else if (!newSig.name || newSig.name === '') {
|
||||||
state = 1;
|
state = 1;
|
||||||
} else if (!!newSig.group && newSig.group !== '' && newSig.name !== '') {
|
} else if (newSig.name !== '') {
|
||||||
state = 2;
|
state = 2;
|
||||||
}
|
}
|
||||||
return state;
|
return state;
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
export * from './renderIcon';
|
export * from './renderIcon';
|
||||||
export * from './renderName';
|
export * from './renderName';
|
||||||
export * from './renderTimeLeft';
|
export * from './renderTimeLeft';
|
||||||
|
export * from './renderLinkedSystem';
|
||||||
|
export * from './renderInfoColumn';
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
import { GroupType, SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
import { GROUPS } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
import { GROUPS } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
||||||
|
|
||||||
export const renderIcon = (row: SystemSignature) => {
|
export const renderIcon = (row: SystemSignature, customSize?: Omit<GroupType, 'icon' | 'id'>) => {
|
||||||
if (row.group == null) {
|
if (row.group == null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -13,7 +13,7 @@ export const renderIcon = (row: SystemSignature) => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex justify-center items-center">
|
<div className="flex justify-center items-center">
|
||||||
<img src={group.icon} style={{ width: group.w, height: group.h }} />
|
<img src={group.icon} style={{ width: customSize?.w ?? group.w, height: customSize?.h ?? group.h }} />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -0,0 +1,3 @@
|
|||||||
|
.whFontSize {
|
||||||
|
font-size: 11px !important;
|
||||||
|
}
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
import { SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { SystemViewStandalone, WHClassView } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { renderName } from './renderName.tsx';
|
||||||
|
import classes from './renderInfoColumn.module.scss';
|
||||||
|
|
||||||
|
export const renderInfoColumn = (row: SystemSignature) => {
|
||||||
|
if (!row.group || row.group === SignatureGroup.Wormhole) {
|
||||||
|
return (
|
||||||
|
<div className="flex justify-start items-center gap-[6px]">
|
||||||
|
{row.type && (
|
||||||
|
<WHClassView
|
||||||
|
className="text-[11px]"
|
||||||
|
classNameWh={classes.whFontSize}
|
||||||
|
highlightName
|
||||||
|
hideWhClass={!!row.linked_system}
|
||||||
|
whClassName={row.type}
|
||||||
|
noOffset
|
||||||
|
useShortTitle
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{row.linked_system && (
|
||||||
|
<>
|
||||||
|
{/*<span className="w-4 h-4 hero-arrow-long-right"></span>*/}
|
||||||
|
<span title={row.linked_system?.solar_system_name}>
|
||||||
|
<SystemViewStandalone
|
||||||
|
className={clsx('select-none text-center cursor-context-menu')}
|
||||||
|
hideRegion
|
||||||
|
{...row.linked_system}
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (row.description != null && row.description.length > 0) {
|
||||||
|
return <span title={row.description}>{row.description}</span>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return renderName(row);
|
||||||
|
};
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
import clsx from 'clsx';
|
||||||
|
|
||||||
|
import { SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { SystemViewStandalone } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
|
export const renderLinkedSystem = (row: SystemSignature) => {
|
||||||
|
if (!row.linked_system) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<span title={row.linked_system?.solar_system_name}>
|
||||||
|
<SystemViewStandalone
|
||||||
|
className={clsx('select-none text-center cursor-context-menu')}
|
||||||
|
hideRegion
|
||||||
|
{...row.linked_system}
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -3,7 +3,7 @@ import { TimeLeft } from '@/hooks/Mapper/components/ui-kit';
|
|||||||
|
|
||||||
export const renderTimeLeft = (row: SystemSignature) => {
|
export const renderTimeLeft = (row: SystemSignature) => {
|
||||||
return (
|
return (
|
||||||
<div className="flex justify-end w-full items-center">
|
<div className="flex w-full items-center">
|
||||||
<TimeLeft cDate={row.updated_at ? new Date(row.updated_at) : undefined} />
|
<TimeLeft cDate={row.updated_at ? new Date(row.updated_at) : undefined} />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
|||||||
import { useCallback, useState } from 'react';
|
import { useCallback, useState } from 'react';
|
||||||
import { OnTheMap, RightBar } from '@/hooks/Mapper/components/mapRootContent/components';
|
import { OnTheMap, RightBar } from '@/hooks/Mapper/components/mapRootContent/components';
|
||||||
import { MapContextMenu } from '@/hooks/Mapper/components/mapRootContent/components/MapContextMenu/MapContextMenu.tsx';
|
import { MapContextMenu } from '@/hooks/Mapper/components/mapRootContent/components/MapContextMenu/MapContextMenu.tsx';
|
||||||
|
import { useSkipContextMenu } from '@/hooks/Mapper/hooks/useSkipContextMenu';
|
||||||
|
import { MapSettings } from "@/hooks/Mapper/components/mapRootContent/components/MapSettings";
|
||||||
|
|
||||||
export interface MapRootContentProps {}
|
export interface MapRootContentProps {}
|
||||||
|
|
||||||
@@ -15,9 +17,13 @@ export const MapRootContent = ({}: MapRootContentProps) => {
|
|||||||
const { isShowMenu } = interfaceSettings;
|
const { isShowMenu } = interfaceSettings;
|
||||||
|
|
||||||
const [showOnTheMap, setShowOnTheMap] = useState(false);
|
const [showOnTheMap, setShowOnTheMap] = useState(false);
|
||||||
|
const [showMapSettings, setShowMapSettings] = useState(false);
|
||||||
const mapInterface = <MapInterface />;
|
const mapInterface = <MapInterface />;
|
||||||
|
|
||||||
const handleShowOnTheMap = useCallback(() => setShowOnTheMap(true), []);
|
const handleShowOnTheMap = useCallback(() => setShowOnTheMap(true), []);
|
||||||
|
const handleShowMapSettings = useCallback(() => setShowMapSettings(true), []);
|
||||||
|
|
||||||
|
useSkipContextMenu();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Layout map={<MapWrapper refn={mapRef} />}>
|
<Layout map={<MapWrapper refn={mapRef} />}>
|
||||||
@@ -28,18 +34,19 @@ export const MapRootContent = ({}: MapRootContentProps) => {
|
|||||||
{mapInterface}
|
{mapInterface}
|
||||||
</div>
|
</div>
|
||||||
<div className="absolute top-0 right-0 w-14 h-[calc(100%+3.5rem)] pointer-events-auto">
|
<div className="absolute top-0 right-0 w-14 h-[calc(100%+3.5rem)] pointer-events-auto">
|
||||||
<RightBar onShowOnTheMap={handleShowOnTheMap} />
|
<RightBar onShowOnTheMap={handleShowOnTheMap} onShowMapSettings={handleShowMapSettings} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
) : (
|
) : (
|
||||||
<div className="absolute top-0 left-14 w-[calc(100%-3.5rem)] h-[calc(100%-3.5rem)] pointer-events-none">
|
<div className="absolute top-0 left-14 w-[calc(100%-3.5rem)] h-[calc(100%-3.5rem)] pointer-events-none">
|
||||||
<Topbar>
|
<Topbar>
|
||||||
<MapContextMenu onShowOnTheMap={handleShowOnTheMap} />
|
<MapContextMenu onShowOnTheMap={handleShowOnTheMap} onShowMapSettings={handleShowMapSettings} />
|
||||||
</Topbar>
|
</Topbar>
|
||||||
{mapInterface}
|
{mapInterface}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
<OnTheMap show={showOnTheMap} onHide={() => setShowOnTheMap(false)} />
|
<OnTheMap show={showOnTheMap} onHide={() => setShowOnTheMap(false)} />
|
||||||
|
<MapSettings show={showMapSettings} onHide={() => setShowMapSettings(false)} />
|
||||||
</Layout>
|
</Layout>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,13 +1,20 @@
|
|||||||
import classes from './Connections.module.scss';
|
import classes from './Connections.module.scss';
|
||||||
import { Sidebar } from 'primereact/sidebar';
|
import { Sidebar } from 'primereact/sidebar';
|
||||||
import { useEffect, useMemo, useState } from 'react';
|
import { useEffect, useMemo, useState, useCallback } from 'react';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { VirtualScroller, VirtualScrollerTemplateOptions } from 'primereact/virtualscroller';
|
import { VirtualScroller, VirtualScrollerTemplateOptions } from 'primereact/virtualscroller';
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import { ConnectionOutput, OutCommand, Passage, SolarSystemConnection } from '@/hooks/Mapper/types';
|
import {
|
||||||
|
ConnectionOutput,
|
||||||
|
ConnectionInfoOutput,
|
||||||
|
OutCommand,
|
||||||
|
Passage,
|
||||||
|
SolarSystemConnection,
|
||||||
|
} from '@/hooks/Mapper/types';
|
||||||
import { PassageCard } from './PassageCard';
|
import { PassageCard } from './PassageCard';
|
||||||
import { InfoDrawer, SystemView } from '@/hooks/Mapper/components/ui-kit';
|
import { InfoDrawer, SystemView } from '@/hooks/Mapper/components/ui-kit';
|
||||||
import { kgToTons } from '@/hooks/Mapper/utils/kgToTons.ts';
|
import { kgToTons } from '@/hooks/Mapper/utils/kgToTons.ts';
|
||||||
|
import { TimeAgo } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
const sortByDate = (a: string, b: string) => new Date(a).getTime() - new Date(b).getTime();
|
const sortByDate = (a: string, b: string) => new Date(a).getTime() - new Date(b).getTime();
|
||||||
|
|
||||||
@@ -69,25 +76,44 @@ export const Connections = ({ selectedConnection, onHide }: OnTheMapProps) => {
|
|||||||
}, [connections, selectedConnection]);
|
}, [connections, selectedConnection]);
|
||||||
|
|
||||||
const [passages, setPassages] = useState<Passage[]>([]);
|
const [passages, setPassages] = useState<Passage[]>([]);
|
||||||
|
const [info, setInfo] = useState<ConnectionInfoOutput>(null);
|
||||||
|
|
||||||
|
const loadInfo = useCallback(
|
||||||
|
async (connection: SolarSystemConnection) => {
|
||||||
|
const result = await outCommand<ConnectionInfoOutput>({
|
||||||
|
type: OutCommand.getConnectionInfo,
|
||||||
|
data: {
|
||||||
|
from: connection.source,
|
||||||
|
to: connection.target,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
setInfo(result);
|
||||||
|
},
|
||||||
|
[outCommand],
|
||||||
|
);
|
||||||
|
|
||||||
|
const loadPassages = useCallback(
|
||||||
|
async (connection: SolarSystemConnection) => {
|
||||||
|
const result = await outCommand<ConnectionOutput>({
|
||||||
|
type: OutCommand.getPassages,
|
||||||
|
data: {
|
||||||
|
from: connection.source,
|
||||||
|
to: connection.target,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
setPassages(result.passages.sort((a, b) => sortByDate(b.inserted_at, a.inserted_at)));
|
||||||
|
},
|
||||||
|
[outCommand],
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!selectedConnection) {
|
if (!selectedConnection) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
loadInfo(selectedConnection);
|
||||||
const loadInfo = async () => {
|
loadPassages(selectedConnection);
|
||||||
const result = await outCommand<ConnectionOutput>({
|
|
||||||
type: OutCommand.getPassages,
|
|
||||||
data: {
|
|
||||||
from: selectedConnection.source,
|
|
||||||
to: selectedConnection.target,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
setPassages(result.passages.sort((a, b) => sortByDate(b.inserted_at, a.inserted_at)));
|
|
||||||
};
|
|
||||||
|
|
||||||
loadInfo();
|
|
||||||
}, [selectedConnection]);
|
}, [selectedConnection]);
|
||||||
|
|
||||||
const approximateMass = useMemo(() => {
|
const approximateMass = useMemo(() => {
|
||||||
@@ -132,6 +158,10 @@ export const Connections = ({ selectedConnection, onHide }: OnTheMapProps) => {
|
|||||||
{kgToTons(approximateMass)}
|
{kgToTons(approximateMass)}
|
||||||
</InfoDrawer>
|
</InfoDrawer>
|
||||||
|
|
||||||
|
<InfoDrawer title="Mark EOL Time" rightSide>
|
||||||
|
{info?.marl_eol_time ? <TimeAgo timestamp={info.marl_eol_time} /> : ' unknown '}
|
||||||
|
</InfoDrawer>
|
||||||
|
|
||||||
<div className="flex gap-2"></div>
|
<div className="flex gap-2"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -8,10 +8,11 @@ import { MenuItem } from 'primereact/menuitem';
|
|||||||
|
|
||||||
export interface MapContextMenuProps {
|
export interface MapContextMenuProps {
|
||||||
onShowOnTheMap?: () => void;
|
onShowOnTheMap?: () => void;
|
||||||
|
onShowMapSettings?: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const MapContextMenu = ({ onShowOnTheMap }: MapContextMenuProps) => {
|
export const MapContextMenu = ({ onShowOnTheMap, onShowMapSettings }: MapContextMenuProps) => {
|
||||||
const { outCommand, interfaceSettings, setInterfaceSettings } = useMapRootState();
|
const { outCommand, setInterfaceSettings } = useMapRootState();
|
||||||
|
|
||||||
const menuRight = useRef<Menu>(null);
|
const menuRight = useRef<Menu>(null);
|
||||||
|
|
||||||
@@ -22,13 +23,6 @@ export const MapContextMenu = ({ onShowOnTheMap }: MapContextMenuProps) => {
|
|||||||
});
|
});
|
||||||
}, [outCommand]);
|
}, [outCommand]);
|
||||||
|
|
||||||
const toggleMinimap = useCallback(() => {
|
|
||||||
setInterfaceSettings(x => ({
|
|
||||||
...x,
|
|
||||||
isShowMinimap: !x.isShowMinimap,
|
|
||||||
}));
|
|
||||||
}, [setInterfaceSettings]);
|
|
||||||
|
|
||||||
const items = useMemo(() => {
|
const items = useMemo(() => {
|
||||||
return [
|
return [
|
||||||
{
|
{
|
||||||
@@ -43,9 +37,9 @@ export const MapContextMenu = ({ onShowOnTheMap }: MapContextMenuProps) => {
|
|||||||
},
|
},
|
||||||
{ separator: true },
|
{ separator: true },
|
||||||
{
|
{
|
||||||
label: interfaceSettings.isShowMinimap ? 'Hide minimap' : 'Show minimap',
|
label: 'Settings',
|
||||||
icon: `pi ${interfaceSettings.isShowMinimap ? 'pi-eye-slash' : 'pi-eye'}`,
|
icon: `pi pi-cog`,
|
||||||
command: toggleMinimap,
|
command: onShowMapSettings,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
label: 'Dock menu',
|
label: 'Dock menu',
|
||||||
@@ -57,7 +51,7 @@ export const MapContextMenu = ({ onShowOnTheMap }: MapContextMenuProps) => {
|
|||||||
})),
|
})),
|
||||||
},
|
},
|
||||||
] as MenuItem[];
|
] as MenuItem[];
|
||||||
}, [handleAddCharacter, interfaceSettings.isShowMinimap, onShowOnTheMap, setInterfaceSettings, toggleMinimap]);
|
}, [handleAddCharacter, onShowMapSettings, onShowOnTheMap, setInterfaceSettings]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="ml-1">
|
<div className="ml-1">
|
||||||
|
|||||||
@@ -0,0 +1,127 @@
|
|||||||
|
.verticalTabsContainer {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
min-height: 300px;
|
||||||
|
|
||||||
|
:global {
|
||||||
|
.p-tabview {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-tabview-panels {
|
||||||
|
padding: 6px 1rem !important;
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.CheckboxContainer {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: auto 1fr auto;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
& > span:nth-child(1) {
|
||||||
|
color: var(--gray-200);
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
& > :nth-child(2){
|
||||||
|
border-bottom: 2px dotted #3f3f3f;
|
||||||
|
height: 2px;
|
||||||
|
margin: 0 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Уменьшение размеров InputSwitch с использованием глобальных стилей */
|
||||||
|
.smallInputSwitch {
|
||||||
|
height: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
:global {
|
||||||
|
.p-inputswitch {
|
||||||
|
height: 1rem;
|
||||||
|
width: 2rem;
|
||||||
|
&.p-inputswitch-checked {
|
||||||
|
.p-inputswitch-slider::before {
|
||||||
|
transform: translateX(1rem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.p-highlight .p-inputswitch-slider:before {
|
||||||
|
transform: translateX(1rem);
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-inputswitch-slider::before {
|
||||||
|
width: 0.8rem;
|
||||||
|
height: 0.8rem;
|
||||||
|
margin-top: -0.4rem;
|
||||||
|
margin-left: -3px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,175 @@
|
|||||||
|
import styles from './MapSettings.module.scss';
|
||||||
|
import { Dialog } from 'primereact/dialog';
|
||||||
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
import { TabPanel, TabView } from 'primereact/tabview';
|
||||||
|
import { PrettySwitchbox } from './components';
|
||||||
|
import { InterfaceStoredSettings, InterfaceStoredSettingsProps, useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
|
export enum UserSettingsRemoteProps {
|
||||||
|
link_signature_on_splash = 'link_signature_on_splash',
|
||||||
|
select_on_spash = 'select_on_spash',
|
||||||
|
delete_connection_with_sigs = 'delete_connection_with_sigs',
|
||||||
|
}
|
||||||
|
|
||||||
|
export const DEFAULT_REMOTE_SETTINGS = {
|
||||||
|
[UserSettingsRemoteProps.link_signature_on_splash]: false,
|
||||||
|
[UserSettingsRemoteProps.select_on_spash]: false,
|
||||||
|
[UserSettingsRemoteProps.delete_connection_with_sigs]: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const UserSettingsRemoteList = [
|
||||||
|
UserSettingsRemoteProps.link_signature_on_splash,
|
||||||
|
UserSettingsRemoteProps.select_on_spash,
|
||||||
|
UserSettingsRemoteProps.delete_connection_with_sigs,
|
||||||
|
];
|
||||||
|
|
||||||
|
export type UserSettingsRemote = {
|
||||||
|
link_signature_on_splash: boolean;
|
||||||
|
select_on_spash: boolean;
|
||||||
|
delete_connection_with_sigs: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type UserSettings = UserSettingsRemote & InterfaceStoredSettings;
|
||||||
|
|
||||||
|
export interface MapSettingsProps {
|
||||||
|
show: boolean;
|
||||||
|
onHide: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
type CheckboxesList = {
|
||||||
|
prop: keyof UserSettings;
|
||||||
|
label: string;
|
||||||
|
}[];
|
||||||
|
|
||||||
|
const COMMON_CHECKBOXES_PROPS: CheckboxesList = [
|
||||||
|
{ prop: InterfaceStoredSettingsProps.isShowMinimap, label: 'Show Minimap' },
|
||||||
|
];
|
||||||
|
|
||||||
|
const SYSTEMS_CHECKBOXES_PROPS: CheckboxesList = [
|
||||||
|
{ prop: InterfaceStoredSettingsProps.isShowKSpace, label: 'Highlight Low/High-security systems' },
|
||||||
|
{ prop: UserSettingsRemoteProps.select_on_spash, label: 'Auto-select splashed' },
|
||||||
|
];
|
||||||
|
|
||||||
|
const SIGNATURES_CHECKBOXES_PROPS: CheckboxesList = [
|
||||||
|
{ prop: UserSettingsRemoteProps.link_signature_on_splash, label: 'Link signature on splash' },
|
||||||
|
];
|
||||||
|
|
||||||
|
const CONNECTIONS_CHECKBOXES_PROPS: CheckboxesList = [
|
||||||
|
{ prop: UserSettingsRemoteProps.delete_connection_with_sigs, label: 'Delete connections to linked signatures' },
|
||||||
|
];
|
||||||
|
|
||||||
|
const UI_CHECKBOXES_PROPS: CheckboxesList = [
|
||||||
|
{ prop: InterfaceStoredSettingsProps.isShowMenu, label: 'Enable compact map menu bar' },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const MapSettings = ({ show, onHide }: MapSettingsProps) => {
|
||||||
|
const [activeIndex, setActiveIndex] = useState(0);
|
||||||
|
const { outCommand, interfaceSettings, setInterfaceSettings } = useMapRootState();
|
||||||
|
const [userRemoteSettings, setUserRemoteSettings] = useState<UserSettingsRemote>({ ...DEFAULT_REMOTE_SETTINGS });
|
||||||
|
|
||||||
|
const mergedSettings = useMemo(() => {
|
||||||
|
return {
|
||||||
|
...interfaceSettings,
|
||||||
|
...userRemoteSettings,
|
||||||
|
};
|
||||||
|
}, [userRemoteSettings, interfaceSettings]);
|
||||||
|
|
||||||
|
const handleShow = async () => {
|
||||||
|
const { user_settings } = await outCommand({
|
||||||
|
type: OutCommand.getUserSettings,
|
||||||
|
data: null,
|
||||||
|
});
|
||||||
|
|
||||||
|
setUserRemoteSettings({
|
||||||
|
...user_settings,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleChangeChecked = useCallback(
|
||||||
|
(prop: keyof UserSettings) => async (checked: boolean) => {
|
||||||
|
// @ts-ignore
|
||||||
|
if (UserSettingsRemoteList.includes(prop)) {
|
||||||
|
const newRemoteSettings = {
|
||||||
|
...userRemoteSettings,
|
||||||
|
[prop]: checked,
|
||||||
|
};
|
||||||
|
|
||||||
|
await outCommand({
|
||||||
|
type: OutCommand.updateUserSettings,
|
||||||
|
data: newRemoteSettings,
|
||||||
|
});
|
||||||
|
|
||||||
|
setUserRemoteSettings(newRemoteSettings);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setInterfaceSettings({
|
||||||
|
...interfaceSettings,
|
||||||
|
[prop]: checked,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
[interfaceSettings, outCommand, setInterfaceSettings, userRemoteSettings],
|
||||||
|
);
|
||||||
|
|
||||||
|
const renderCheckboxesList = (list: CheckboxesList) => {
|
||||||
|
return list.map(x => {
|
||||||
|
return (
|
||||||
|
<PrettySwitchbox
|
||||||
|
key={x.prop}
|
||||||
|
label={x.label}
|
||||||
|
checked={mergedSettings[x.prop]}
|
||||||
|
setChecked={handleChangeChecked(x.prop)}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog
|
||||||
|
header="Map settings"
|
||||||
|
visible={show}
|
||||||
|
draggable={false}
|
||||||
|
style={{ width: '550px' }}
|
||||||
|
onShow={handleShow}
|
||||||
|
onHide={() => {
|
||||||
|
if (!show) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setActiveIndex(0);
|
||||||
|
onHide();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="flex flex-col gap-3">
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
<div className={styles.verticalTabsContainer}>
|
||||||
|
<TabView
|
||||||
|
activeIndex={activeIndex}
|
||||||
|
onTabChange={e => setActiveIndex(e.index)}
|
||||||
|
className={styles.verticalTabView}
|
||||||
|
>
|
||||||
|
<TabPanel header="Common" headerClassName={styles.verticalTabHeader}>
|
||||||
|
<div className="w-full h-full flex flex-col gap-1">{renderCheckboxesList(COMMON_CHECKBOXES_PROPS)}</div>
|
||||||
|
</TabPanel>
|
||||||
|
<TabPanel header="Systems" headerClassName={styles.verticalTabHeader}>
|
||||||
|
<div className="w-full h-full flex flex-col gap-1">
|
||||||
|
{renderCheckboxesList(SYSTEMS_CHECKBOXES_PROPS)}
|
||||||
|
</div>
|
||||||
|
</TabPanel>
|
||||||
|
<TabPanel header="Connections" headerClassName={styles.verticalTabHeader}>
|
||||||
|
{renderCheckboxesList(CONNECTIONS_CHECKBOXES_PROPS)}
|
||||||
|
</TabPanel>
|
||||||
|
<TabPanel header="Signatures" headerClassName={styles.verticalTabHeader}>
|
||||||
|
{renderCheckboxesList(SIGNATURES_CHECKBOXES_PROPS)}
|
||||||
|
</TabPanel>
|
||||||
|
<TabPanel header="User Interface" headerClassName={styles.verticalTabHeader}>
|
||||||
|
{renderCheckboxesList(UI_CHECKBOXES_PROPS)}
|
||||||
|
</TabPanel>
|
||||||
|
</TabView>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
.CheckboxContainer {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: auto 1fr auto;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
& > span:nth-child(1) {
|
||||||
|
color: var(--gray-200);
|
||||||
|
font-size: 13px;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
& > :nth-child(2){
|
||||||
|
border-bottom: 2px dotted #3f3f3f;
|
||||||
|
height: 1px;
|
||||||
|
margin: 0 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Уменьшение размеров InputSwitch с использованием глобальных стилей */
|
||||||
|
.smallInputSwitch {
|
||||||
|
height: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
:global {
|
||||||
|
.p-inputswitch {
|
||||||
|
height: 1rem;
|
||||||
|
width: 2rem;
|
||||||
|
&.p-inputswitch-checked {
|
||||||
|
.p-inputswitch-slider::before {
|
||||||
|
transform: translateX(1rem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.p-highlight .p-inputswitch-slider:before {
|
||||||
|
transform: translateX(1rem);
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-inputswitch-slider::before {
|
||||||
|
width: 0.8rem;
|
||||||
|
height: 0.8rem;
|
||||||
|
margin-top: -0.4rem;
|
||||||
|
margin-left: -3px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
import styles from './MapSettings.module.scss';
|
||||||
|
import { WdCheckbox } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
|
interface PrettySwitchboxProps {
|
||||||
|
checked: boolean;
|
||||||
|
setChecked: (checked: boolean) => void;
|
||||||
|
label: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const PrettySwitchbox = ({ checked, setChecked, label }: PrettySwitchboxProps) => {
|
||||||
|
return (
|
||||||
|
<label className={styles.CheckboxContainer}>
|
||||||
|
<span>{label}</span>
|
||||||
|
<div />
|
||||||
|
<div className={styles.smallInputSwitch}>
|
||||||
|
<WdCheckbox size="m" label={''} value={checked} onChange={e => setChecked(e.checked ?? false)} />
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './PrettySwitchbox';
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './PrettySwitchbox';
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './MapSettings';
|
||||||
@@ -8,11 +8,14 @@ import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
|||||||
|
|
||||||
interface RightBarProps {
|
interface RightBarProps {
|
||||||
onShowOnTheMap?: () => void;
|
onShowOnTheMap?: () => void;
|
||||||
|
onShowMapSettings?: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const RightBar = ({ onShowOnTheMap }: RightBarProps) => {
|
export const RightBar = ({ onShowOnTheMap, onShowMapSettings }: RightBarProps) => {
|
||||||
const { outCommand, interfaceSettings, setInterfaceSettings } = useMapRootState();
|
const { outCommand, interfaceSettings, setInterfaceSettings } = useMapRootState();
|
||||||
|
|
||||||
|
const isShowMinimap = interfaceSettings.isShowMinimap === undefined ? true : interfaceSettings.isShowMinimap;
|
||||||
|
|
||||||
const handleAddCharacter = useCallback(() => {
|
const handleAddCharacter = useCallback(() => {
|
||||||
outCommand({
|
outCommand({
|
||||||
type: OutCommand.addCharacter,
|
type: OutCommand.addCharacter,
|
||||||
@@ -27,6 +30,13 @@ export const RightBar = ({ onShowOnTheMap }: RightBarProps) => {
|
|||||||
}));
|
}));
|
||||||
}, [setInterfaceSettings]);
|
}, [setInterfaceSettings]);
|
||||||
|
|
||||||
|
const toggleKSpace = useCallback(() => {
|
||||||
|
setInterfaceSettings(x => ({
|
||||||
|
...x,
|
||||||
|
isShowKSpace: !x.isShowKSpace,
|
||||||
|
}));
|
||||||
|
}, [setInterfaceSettings]);
|
||||||
|
|
||||||
const toggleMenu = useCallback(() => {
|
const toggleMenu = useCallback(() => {
|
||||||
setInterfaceSettings(x => ({
|
setInterfaceSettings(x => ({
|
||||||
...x,
|
...x,
|
||||||
@@ -50,7 +60,7 @@ export const RightBar = ({ onShowOnTheMap }: RightBarProps) => {
|
|||||||
type="button"
|
type="button"
|
||||||
onClick={handleAddCharacter}
|
onClick={handleAddCharacter}
|
||||||
>
|
>
|
||||||
<i className="pi pi-user-plus text-lg"></i>
|
<i className="pi pi-user-plus"></i>
|
||||||
</button>
|
</button>
|
||||||
</WdTooltipWrapper>
|
</WdTooltipWrapper>
|
||||||
|
|
||||||
@@ -60,26 +70,44 @@ export const RightBar = ({ onShowOnTheMap }: RightBarProps) => {
|
|||||||
type="button"
|
type="button"
|
||||||
onClick={onShowOnTheMap}
|
onClick={onShowOnTheMap}
|
||||||
>
|
>
|
||||||
<i className="pi pi-hashtag text-lg"></i>
|
<i className="pi pi-hashtag"></i>
|
||||||
</button>
|
</button>
|
||||||
</WdTooltipWrapper>
|
</WdTooltipWrapper>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-col items-center mb-2 gap-1">
|
<div className="flex flex-col items-center mb-2 gap-1">
|
||||||
|
<WdTooltipWrapper content="User settings" position={TooltipPosition.left}>
|
||||||
|
<button
|
||||||
|
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
|
||||||
|
type="button"
|
||||||
|
onClick={onShowMapSettings}
|
||||||
|
>
|
||||||
|
<i className="pi pi-cog"></i>
|
||||||
|
</button>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
|
||||||
<WdTooltipWrapper
|
<WdTooltipWrapper
|
||||||
content={interfaceSettings.isShowMinimap ? 'Hide minimap' : 'Show minimap'}
|
content={
|
||||||
|
interfaceSettings.isShowKSpace ? 'Hide highlighting Imperial Space' : 'Show highlighting Imperial Space'
|
||||||
|
}
|
||||||
position={TooltipPosition.left}
|
position={TooltipPosition.left}
|
||||||
>
|
>
|
||||||
|
<button
|
||||||
|
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
|
||||||
|
type="button"
|
||||||
|
onClick={toggleKSpace}
|
||||||
|
>
|
||||||
|
<i className={interfaceSettings.isShowKSpace ? 'hero-cloud-solid' : 'hero-cloud'}></i>
|
||||||
|
</button>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
|
||||||
|
<WdTooltipWrapper content={isShowMinimap ? 'Hide minimap' : 'Show minimap'} position={TooltipPosition.left}>
|
||||||
<button
|
<button
|
||||||
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
|
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
|
||||||
type="button"
|
type="button"
|
||||||
onClick={toggleMinimap}
|
onClick={toggleMinimap}
|
||||||
>
|
>
|
||||||
{interfaceSettings.isShowMinimap ? (
|
<i className={isShowMinimap ? 'pi pi-eye' : 'pi pi-eye-slash'}></i>
|
||||||
<i className="pi pi-eye text-lg"></i>
|
|
||||||
) : (
|
|
||||||
<i className="pi pi-eye-slash text-lg"></i>
|
|
||||||
)}
|
|
||||||
</button>
|
</button>
|
||||||
</WdTooltipWrapper>
|
</WdTooltipWrapper>
|
||||||
|
|
||||||
@@ -89,7 +117,7 @@ export const RightBar = ({ onShowOnTheMap }: RightBarProps) => {
|
|||||||
type="button"
|
type="button"
|
||||||
onClick={toggleMenu}
|
onClick={toggleMenu}
|
||||||
>
|
>
|
||||||
<i className="pi pi-window-minimize text-lg"></i>
|
<i className="pi pi-window-minimize"></i>
|
||||||
</button>
|
</button>
|
||||||
</WdTooltipWrapper>
|
</WdTooltipWrapper>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -0,0 +1,10 @@
|
|||||||
|
import { createGenericContext } from '@/hooks/Mapper/utils/abstractContextProvider.tsx';
|
||||||
|
|
||||||
|
export interface SystemsSettingsProvider {
|
||||||
|
systemId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { Provider, useContextValue } = createGenericContext<SystemsSettingsProvider>();
|
||||||
|
|
||||||
|
export const SystemsSettingsProvider = Provider;
|
||||||
|
export const useSystemsSettingsProvider = useContextValue;
|
||||||
@@ -0,0 +1,81 @@
|
|||||||
|
.verticalTabsContainer {
|
||||||
|
width: 100%;
|
||||||
|
min-height: 300px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.verticalTabsContainer {
|
||||||
|
display: flex;
|
||||||
|
width: 100%;
|
||||||
|
min-height: 300px;
|
||||||
|
|
||||||
|
:global {
|
||||||
|
.p-tabview {
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-tabview-panels {
|
||||||
|
padding: 6px 1rem !important;
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.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: 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 {
|
||||||
|
border-right: 4px solid var(--primary-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-tabview-panel {
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,171 @@
|
|||||||
|
import { Dialog } from 'primereact/dialog';
|
||||||
|
import { useCallback, useEffect } from 'react';
|
||||||
|
// import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { OutCommand, SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { Controller, FormProvider, useForm } from 'react-hook-form';
|
||||||
|
import {
|
||||||
|
SignatureGroupContent,
|
||||||
|
SignatureGroupSelect,
|
||||||
|
} from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components';
|
||||||
|
import { InputText } from 'primereact/inputtext';
|
||||||
|
import { SystemsSettingsProvider } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/Provider.tsx';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
|
type SystemSignaturePrepared = Omit<SystemSignature, 'linked_system'> & { linked_system: string };
|
||||||
|
|
||||||
|
export interface MapSettingsProps {
|
||||||
|
systemId: string;
|
||||||
|
show: boolean;
|
||||||
|
onHide: () => void;
|
||||||
|
signatureData: SystemSignature | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SignatureSettings = ({ systemId, show, onHide, signatureData }: MapSettingsProps) => {
|
||||||
|
const { outCommand } = useMapRootState();
|
||||||
|
|
||||||
|
const handleShow = async () => {};
|
||||||
|
const form = useForm<Partial<SystemSignaturePrepared>>({});
|
||||||
|
|
||||||
|
const handleSave = useCallback(async () => {
|
||||||
|
if (!signatureData) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { group, ...values } = form.getValues();
|
||||||
|
let out = { ...signatureData };
|
||||||
|
|
||||||
|
switch (group) {
|
||||||
|
case SignatureGroup.Wormhole:
|
||||||
|
if (values.linked_system) {
|
||||||
|
await outCommand({
|
||||||
|
type: OutCommand.linkSignatureToSystem,
|
||||||
|
data: {
|
||||||
|
signature_eve_id: signatureData.eve_id,
|
||||||
|
solar_system_source: systemId,
|
||||||
|
solar_system_target: values.linked_system,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (values.type != null) {
|
||||||
|
out = { ...out, type: values.type };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (signatureData.group !== SignatureGroup.Wormhole) {
|
||||||
|
out = { ...out, name: '' };
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
case SignatureGroup.CosmicSignature:
|
||||||
|
out = { ...out, type: '', name: '' };
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
if (values.name != null) {
|
||||||
|
out = { ...out, name: values.name ?? '' };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (values.description != null) {
|
||||||
|
out = { ...out, description: values.description };
|
||||||
|
}
|
||||||
|
|
||||||
|
// Note: when type of signature changed from WH to other type - we should drop name
|
||||||
|
if (
|
||||||
|
group !== SignatureGroup.Wormhole && // new
|
||||||
|
signatureData.group === SignatureGroup.Wormhole && // prev
|
||||||
|
signatureData.linked_system
|
||||||
|
) {
|
||||||
|
await outCommand({
|
||||||
|
type: OutCommand.unlinkSignature,
|
||||||
|
data: { signature_eve_id: signatureData.eve_id, solar_system_source: systemId },
|
||||||
|
});
|
||||||
|
|
||||||
|
out = { ...out, type: '' };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (group === SignatureGroup.Wormhole && signatureData.linked_system != null && values.linked_system === null) {
|
||||||
|
await outCommand({
|
||||||
|
type: OutCommand.unlinkSignature,
|
||||||
|
data: { signature_eve_id: signatureData.eve_id, solar_system_source: systemId },
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Note: despite groups have optional type - this will always set
|
||||||
|
out = { ...out, group: group! };
|
||||||
|
|
||||||
|
await outCommand({
|
||||||
|
type: OutCommand.updateSignatures,
|
||||||
|
data: {
|
||||||
|
system_id: systemId,
|
||||||
|
added: [],
|
||||||
|
updated: [out],
|
||||||
|
removed: [],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
form.reset();
|
||||||
|
onHide();
|
||||||
|
}, [form, onHide, outCommand, signatureData, systemId]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!signatureData) {
|
||||||
|
form.reset();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { linked_system, ...rest } = signatureData;
|
||||||
|
|
||||||
|
form.reset({
|
||||||
|
linked_system: linked_system?.solar_system_id.toString() ?? undefined,
|
||||||
|
...rest,
|
||||||
|
});
|
||||||
|
}, [form, signatureData]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog
|
||||||
|
header={`Signature Edit [${signatureData?.eve_id}]`}
|
||||||
|
visible={show}
|
||||||
|
draggable={false}
|
||||||
|
style={{ width: '390px' }}
|
||||||
|
onShow={handleShow}
|
||||||
|
onHide={() => {
|
||||||
|
if (!show) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
onHide();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<SystemsSettingsProvider initialValue={{ systemId }}>
|
||||||
|
<FormProvider {...form}>
|
||||||
|
<div className="flex flex-col gap-2 justify-between">
|
||||||
|
<div className="w-full flex flex-col gap-1 p-1 min-h-[150px]">
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center text-[14px]">
|
||||||
|
<span>Group:</span>
|
||||||
|
<SignatureGroupSelect name="group" />
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<SignatureGroupContent />
|
||||||
|
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center text-[14px]">
|
||||||
|
<span>Description:</span>
|
||||||
|
<Controller
|
||||||
|
name="description"
|
||||||
|
control={form.control}
|
||||||
|
render={({ field }) => (
|
||||||
|
<InputText placeholder="Type description" value={field.value} onChange={field.onChange} />
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex gap-2 justify-end">
|
||||||
|
<Button onClick={handleSave} outlined size="small" label="Save"></Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</FormProvider>
|
||||||
|
</SystemsSettingsProvider>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,45 @@
|
|||||||
|
import { Controller, useFormContext } from 'react-hook-form';
|
||||||
|
import { SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { useSystemsSettingsProvider } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/Provider.tsx';
|
||||||
|
import { SignatureGroupContentWormholes } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureGroupContentWormholes.tsx';
|
||||||
|
import { InputText } from 'primereact/inputtext';
|
||||||
|
|
||||||
|
export interface SignatureGroupContentProps {}
|
||||||
|
|
||||||
|
export const SignatureGroupContent = ({}: SignatureGroupContentProps) => {
|
||||||
|
const { watch, control } = useFormContext<SystemSignature>();
|
||||||
|
const group = watch('group');
|
||||||
|
|
||||||
|
const {
|
||||||
|
value: { systemId },
|
||||||
|
} = useSystemsSettingsProvider();
|
||||||
|
|
||||||
|
if (!systemId) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (group === SignatureGroup.Wormhole) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<SignatureGroupContentWormholes />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (group === SignatureGroup.CosmicSignature) {
|
||||||
|
return <div></div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center text-[14px]">
|
||||||
|
<span>Name:</span>
|
||||||
|
<Controller
|
||||||
|
name="name"
|
||||||
|
control={control}
|
||||||
|
render={({ field }) => <InputText placeholder="Name" value={field.value} onChange={field.onChange} />}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SignatureGroupContent';
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
import { SignatureWormholeTypeSelect } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureWormholeTypeSelect';
|
||||||
|
import { SignatureLeadsToSelect } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureLeadsToSelect';
|
||||||
|
|
||||||
|
export const SignatureGroupContentWormholes = () => {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center text-[14px]">
|
||||||
|
<span>Type:</span>
|
||||||
|
<SignatureWormholeTypeSelect name="type" />
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center text-[14px]">
|
||||||
|
<span>Leads To:</span>
|
||||||
|
<SignatureLeadsToSelect name="linked_system" />
|
||||||
|
</label>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,59 @@
|
|||||||
|
import { Dropdown } from 'primereact/dropdown';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { renderIcon } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/renders';
|
||||||
|
import { Controller, useFormContext } from 'react-hook-form';
|
||||||
|
|
||||||
|
const signatureGroupOptions = Object.keys(SignatureGroup).map(x => ({
|
||||||
|
value: SignatureGroup[x as keyof typeof SignatureGroup],
|
||||||
|
label: SignatureGroup[x as keyof typeof SignatureGroup],
|
||||||
|
}));
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const renderSignatureTemplate = option => {
|
||||||
|
if (!option) {
|
||||||
|
return 'No group selected';
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<span className="w-[20px] mt-[1px] flex justify-center items-center">
|
||||||
|
{renderIcon(
|
||||||
|
{ group: option.label } as SystemSignature,
|
||||||
|
option.label === SignatureGroup.CosmicSignature ? { w: 10, h: 10 } : { w: 16, h: 16 },
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
<span>{option.label}</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface SignatureGroupSelectProps {
|
||||||
|
name: string;
|
||||||
|
defaultValue?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SignatureGroupSelect = ({ name, defaultValue = '' }: SignatureGroupSelectProps) => {
|
||||||
|
const { control } = useFormContext();
|
||||||
|
return (
|
||||||
|
<Controller
|
||||||
|
name={name}
|
||||||
|
control={control}
|
||||||
|
defaultValue={defaultValue}
|
||||||
|
render={({ field }) => (
|
||||||
|
<Dropdown
|
||||||
|
value={field.value}
|
||||||
|
onChange={field.onChange}
|
||||||
|
options={signatureGroupOptions}
|
||||||
|
optionLabel="label"
|
||||||
|
optionValue="value"
|
||||||
|
placeholder="Select group"
|
||||||
|
className={clsx('w-full')}
|
||||||
|
scrollHeight="240px"
|
||||||
|
itemTemplate={renderSignatureTemplate}
|
||||||
|
valueTemplate={renderSignatureTemplate}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SignatureGroupSelect';
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
.SystemView {
|
||||||
|
font-size: 14px !important;
|
||||||
|
}
|
||||||
@@ -0,0 +1,108 @@
|
|||||||
|
import { Dropdown } from 'primereact/dropdown';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { Controller, useFormContext } from 'react-hook-form';
|
||||||
|
import { useSystemsSettingsProvider } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/Provider.tsx';
|
||||||
|
import { useSystemInfo } from '@/hooks/Mapper/components/hooks';
|
||||||
|
import { useMemo } from 'react';
|
||||||
|
import { SystemView } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import classes from './SignatureLeadsToSelect.module.scss';
|
||||||
|
import { useLoadSystemStatic } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic.ts';
|
||||||
|
import { SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { WORMHOLES_ADDITIONAL_INFO_BY_CLASS_ID } from '@/hooks/Mapper/components/map/constants.ts';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const renderLinkedSystemItem = (option: { value: string }) => {
|
||||||
|
if (option.value == null) {
|
||||||
|
return <div className="flex gap-2 items-center">No linked system</div>;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<SystemView systemId={option.value} className={classes.SystemView} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const renderLinkedSystemValue = (option: { value: string }) => {
|
||||||
|
if (!option) {
|
||||||
|
return 'Select Leads To system';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (option.value == null) {
|
||||||
|
return 'Select Leads To system';
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<SystemView systemId={option.value} className={classes.SystemView} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderLeadsToEmpty = () => <div className="flex items-center text-[14px]">No wormhole to select</div>;
|
||||||
|
|
||||||
|
export interface SignatureLeadsToSelectProps {
|
||||||
|
name: string;
|
||||||
|
defaultValue?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SignatureLeadsToSelect = ({ name, defaultValue = '' }: SignatureLeadsToSelectProps) => {
|
||||||
|
const { control, watch } = useFormContext<SystemSignature>();
|
||||||
|
const group = watch('type');
|
||||||
|
|
||||||
|
const {
|
||||||
|
value: { systemId },
|
||||||
|
} = useSystemsSettingsProvider();
|
||||||
|
|
||||||
|
const { leadsTo } = useSystemInfo({ systemId });
|
||||||
|
const { systems: systemStatics } = useLoadSystemStatic({ systems: leadsTo });
|
||||||
|
const {
|
||||||
|
data: { wormholes },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const leadsToOptions = useMemo(() => {
|
||||||
|
return [
|
||||||
|
{ value: null },
|
||||||
|
...leadsTo
|
||||||
|
.filter(systemId => {
|
||||||
|
const systemStatic = systemStatics.get(parseInt(systemId));
|
||||||
|
const whInfo = wormholes.find(x => x.name === group);
|
||||||
|
|
||||||
|
if (!systemStatic || !whInfo || group === 'K162') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id: whType } = WORMHOLES_ADDITIONAL_INFO_BY_CLASS_ID[systemStatic.system_class];
|
||||||
|
return whInfo.dest === whType;
|
||||||
|
})
|
||||||
|
.map(x => ({ value: x })),
|
||||||
|
];
|
||||||
|
}, [group, leadsTo, systemStatics, wormholes]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Controller
|
||||||
|
// @ts-ignore
|
||||||
|
name={name}
|
||||||
|
control={control}
|
||||||
|
defaultValue={defaultValue}
|
||||||
|
render={({ field }) => {
|
||||||
|
return (
|
||||||
|
<Dropdown
|
||||||
|
value={field.value}
|
||||||
|
onChange={field.onChange}
|
||||||
|
options={leadsToOptions}
|
||||||
|
optionValue="value"
|
||||||
|
placeholder="Select Leads To wormhole"
|
||||||
|
className={clsx('w-full')}
|
||||||
|
scrollHeight="240px"
|
||||||
|
itemTemplate={renderLinkedSystemItem}
|
||||||
|
valueTemplate={renderLinkedSystemValue}
|
||||||
|
emptyMessage={renderLeadsToEmpty}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SignatureLeadsToSelect.tsx';
|
||||||
@@ -0,0 +1,134 @@
|
|||||||
|
import { Dropdown } from 'primereact/dropdown';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { Respawn, SolarSystemStaticInfoRaw, WormholeDataRaw } from '@/hooks/Mapper/types';
|
||||||
|
import { Controller, useFormContext } from 'react-hook-form';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useSystemsSettingsProvider } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/Provider.tsx';
|
||||||
|
import { useSystemInfo } from '@/hooks/Mapper/components/hooks';
|
||||||
|
import {
|
||||||
|
SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS,
|
||||||
|
WORMHOLES_ADDITIONAL_INFO_BY_CLASS_ID,
|
||||||
|
} from '@/hooks/Mapper/components/map/constants.ts';
|
||||||
|
import { useMemo } from 'react';
|
||||||
|
import { WHClassView } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
|
const getPossibleWormholes = (systemStatic: SolarSystemStaticInfoRaw, wormholes: WormholeDataRaw[]) => {
|
||||||
|
const { id: whType } = WORMHOLES_ADDITIONAL_INFO_BY_CLASS_ID[systemStatic.system_class];
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const spawnClassGroup = SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS[whType];
|
||||||
|
const possibleWHTypes = wormholes.filter(x => x.src.includes(spawnClassGroup));
|
||||||
|
|
||||||
|
return {
|
||||||
|
statics: possibleWHTypes
|
||||||
|
.filter(x => x.respawn.some(y => y === Respawn.static))
|
||||||
|
.filter(x => systemStatic.statics.includes(x.name)),
|
||||||
|
k162: wormholes.find(x => x.name === 'K162')!,
|
||||||
|
wanderings: possibleWHTypes.filter(x => x.respawn.some(y => y === Respawn.wandering)),
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const renderWHTypeGroupTemplate = option => {
|
||||||
|
return (
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<span>{option.label}</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const renderWHTypeTemplateValue = (option: { label: string; data: WormholeDataRaw }) => {
|
||||||
|
if (!option) {
|
||||||
|
return 'Select wormhole type';
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<WHClassView whClassName={option.data.name} noOffset useShortTitle />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const renderWHTypeTemplate = (option: { label: string; data: WormholeDataRaw }) => {
|
||||||
|
return (
|
||||||
|
<div className="flex gap-2 items-center ml-[1rem]">
|
||||||
|
<WHClassView whClassName={option.data.name} noOffset useShortTitle />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface SignatureGroupSelectProps {
|
||||||
|
name: string;
|
||||||
|
defaultValue?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SignatureWormholeTypeSelect = ({ name, defaultValue = '' }: SignatureGroupSelectProps) => {
|
||||||
|
const { control } = useFormContext();
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: { wormholes },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const {
|
||||||
|
value: { systemId },
|
||||||
|
} = useSystemsSettingsProvider();
|
||||||
|
|
||||||
|
const system = useSystemInfo({ systemId });
|
||||||
|
|
||||||
|
const possibleWormholesOptions = useMemo(() => {
|
||||||
|
const possibleWormholes = getPossibleWormholes(system.staticInfo, wormholes);
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
label: 'Statics',
|
||||||
|
items: [
|
||||||
|
...possibleWormholes.statics.map(x => ({
|
||||||
|
label: x.name,
|
||||||
|
value: x.name,
|
||||||
|
data: x,
|
||||||
|
})),
|
||||||
|
{
|
||||||
|
value: possibleWormholes.k162.name,
|
||||||
|
label: possibleWormholes.k162.name,
|
||||||
|
data: possibleWormholes.k162,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: 'Wanderings',
|
||||||
|
items: possibleWormholes.wanderings.map(x => ({
|
||||||
|
label: x.name,
|
||||||
|
value: x.name,
|
||||||
|
data: x,
|
||||||
|
})),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}, [system, wormholes]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Controller
|
||||||
|
name={name}
|
||||||
|
control={control}
|
||||||
|
defaultValue={defaultValue}
|
||||||
|
render={({ field }) => (
|
||||||
|
<Dropdown
|
||||||
|
value={field.value}
|
||||||
|
onChange={field.onChange}
|
||||||
|
options={possibleWormholesOptions}
|
||||||
|
optionLabel="label"
|
||||||
|
optionValue="value"
|
||||||
|
placeholder="Select wormhole type"
|
||||||
|
optionGroupLabel="label"
|
||||||
|
optionGroupChildren="items"
|
||||||
|
className={clsx('w-full')}
|
||||||
|
scrollHeight="240px"
|
||||||
|
optionGroupTemplate={renderWHTypeGroupTemplate}
|
||||||
|
itemTemplate={renderWHTypeTemplate}
|
||||||
|
valueTemplate={renderWHTypeTemplateValue}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SignatureWormholeTypeSelect';
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
export * from './SignatureGroupSelect';
|
||||||
|
export * from './SignatureGroupContent';
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SignatureSettings.tsx';
|
||||||
@@ -5,13 +5,20 @@ import { MapRootData, useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
|||||||
import { OnMapSelectionChange } from '@/hooks/Mapper/components/map/map.types.ts';
|
import { OnMapSelectionChange } from '@/hooks/Mapper/components/map/map.types.ts';
|
||||||
import isEqual from 'lodash.isequal';
|
import isEqual from 'lodash.isequal';
|
||||||
import { ContextMenuSystem, useContextMenuSystemHandlers } from '@/hooks/Mapper/components/contexts';
|
import { ContextMenuSystem, useContextMenuSystemHandlers } from '@/hooks/Mapper/components/contexts';
|
||||||
import { SystemCustomLabelDialog, SystemSettingsDialog } from '@/hooks/Mapper/components/mapInterface/components';
|
import {
|
||||||
|
SystemCustomLabelDialog,
|
||||||
|
SystemSettingsDialog,
|
||||||
|
SystemLinkSignatureDialog,
|
||||||
|
} from '@/hooks/Mapper/components/mapInterface/components';
|
||||||
import classes from './MapWrapper.module.scss';
|
import classes from './MapWrapper.module.scss';
|
||||||
import { Connections } from '@/hooks/Mapper/components/mapRootContent/components/Connections';
|
import { Connections } from '@/hooks/Mapper/components/mapRootContent/components/Connections';
|
||||||
import { ContextMenuSystemMultiple, useContextMenuSystemMultipleHandlers } from '../contexts/ContextMenuSystemMultiple';
|
import { ContextMenuSystemMultiple, useContextMenuSystemMultipleHandlers } from '../contexts/ContextMenuSystemMultiple';
|
||||||
import { getSystemById } from '@/hooks/Mapper/helpers';
|
import { getSystemById } from '@/hooks/Mapper/helpers';
|
||||||
import { Node } from 'reactflow';
|
import { Node } from 'reactflow';
|
||||||
|
|
||||||
|
import { Commands } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
import { useMapEventListener } from '@/hooks/Mapper/events';
|
||||||
|
|
||||||
import { STORED_INTERFACE_DEFAULT_VALUES } from '@/hooks/Mapper/mapRootProvider/MapRootProvider';
|
import { STORED_INTERFACE_DEFAULT_VALUES } from '@/hooks/Mapper/mapRootProvider/MapRootProvider';
|
||||||
|
|
||||||
interface MapWrapperProps {
|
interface MapWrapperProps {
|
||||||
@@ -24,7 +31,7 @@ export const MapWrapper = ({ refn }: MapWrapperProps) => {
|
|||||||
update,
|
update,
|
||||||
outCommand,
|
outCommand,
|
||||||
data: { selectedConnections, selectedSystems, hubs, systems },
|
data: { selectedConnections, selectedSystems, hubs, systems },
|
||||||
interfaceSettings: { isShowMenu, isShowMinimap = STORED_INTERFACE_DEFAULT_VALUES.isShowMinimap },
|
interfaceSettings: { isShowMenu, isShowMinimap = STORED_INTERFACE_DEFAULT_VALUES.isShowMinimap, isShowKSpace },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const { open, ...systemContextProps } = useContextMenuSystemHandlers({ systems, hubs, outCommand });
|
const { open, ...systemContextProps } = useContextMenuSystemHandlers({ systems, hubs, outCommand });
|
||||||
@@ -53,6 +60,7 @@ export const MapWrapper = ({ refn }: MapWrapperProps) => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
const [openSettings, setOpenSettings] = useState<string | null>(null);
|
const [openSettings, setOpenSettings] = useState<string | null>(null);
|
||||||
|
const [openLinkSignatures, setOpenLinkSignatures] = useState<any | null>(null);
|
||||||
const [openCustomLabel, setOpenCustomLabel] = useState<string | null>(null);
|
const [openCustomLabel, setOpenCustomLabel] = useState<string | null>(null);
|
||||||
const handleCommand: OutCommandHandler = useCallback(
|
const handleCommand: OutCommandHandler = useCallback(
|
||||||
event => {
|
event => {
|
||||||
@@ -60,6 +68,9 @@ export const MapWrapper = ({ refn }: MapWrapperProps) => {
|
|||||||
case OutCommand.openSettings:
|
case OutCommand.openSettings:
|
||||||
setOpenSettings(event.data.system_id);
|
setOpenSettings(event.data.system_id);
|
||||||
break;
|
break;
|
||||||
|
case OutCommand.linkSignatureToSystem:
|
||||||
|
setOpenLinkSignatures(event.data);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
return outCommand(event);
|
return outCommand(event);
|
||||||
}
|
}
|
||||||
@@ -88,6 +99,14 @@ export const MapWrapper = ({ refn }: MapWrapperProps) => {
|
|||||||
|
|
||||||
const handleConnectionDbClick = useCallback((e: SolarSystemConnection) => setSelectedConnection(e), []);
|
const handleConnectionDbClick = useCallback((e: SolarSystemConnection) => setSelectedConnection(e), []);
|
||||||
|
|
||||||
|
useMapEventListener(event => {
|
||||||
|
switch (event.name) {
|
||||||
|
case Commands.linkSignatureToSystem:
|
||||||
|
setOpenLinkSignatures(event.data);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Map
|
<Map
|
||||||
@@ -99,22 +118,19 @@ export const MapWrapper = ({ refn }: MapWrapperProps) => {
|
|||||||
onSelectionContextMenu={handleSystemMultipleContext}
|
onSelectionContextMenu={handleSystemMultipleContext}
|
||||||
minimapClasses={!isShowMenu ? classes.MiniMap : undefined}
|
minimapClasses={!isShowMenu ? classes.MiniMap : undefined}
|
||||||
isShowMinimap={isShowMinimap}
|
isShowMinimap={isShowMinimap}
|
||||||
|
showKSpaceBG={isShowKSpace}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{openSettings != null && (
|
{openSettings != null && (
|
||||||
<SystemSettingsDialog
|
<SystemSettingsDialog systemId={openSettings} visible setVisible={() => setOpenSettings(null)} />
|
||||||
systemId={openSettings}
|
|
||||||
visible={openSettings != null}
|
|
||||||
setVisible={() => setOpenSettings(null)}
|
|
||||||
/>
|
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{openCustomLabel != null && (
|
{openCustomLabel != null && (
|
||||||
<SystemCustomLabelDialog
|
<SystemCustomLabelDialog systemId={openCustomLabel} visible setVisible={() => setOpenCustomLabel(null)} />
|
||||||
systemId={openCustomLabel}
|
)}
|
||||||
visible={openCustomLabel != null}
|
|
||||||
setVisible={() => setOpenCustomLabel(null)}
|
{openLinkSignatures != null && (
|
||||||
/>
|
<SystemLinkSignatureDialog data={openLinkSignatures} setVisible={() => setOpenLinkSignatures(null)} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Connections selectedConnection={selectedConnection} onHide={() => setSelectedConnection(null)} />
|
<Connections selectedConnection={selectedConnection} onHide={() => setSelectedConnection(null)} />
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ export const CharacterCard = ({
|
|||||||
const { mapRef } = useMapRootState();
|
const { mapRef } = useMapRootState();
|
||||||
|
|
||||||
const handleSelect = useCallback(() => {
|
const handleSelect = useCallback(() => {
|
||||||
mapRef.current?.command(Commands.selectSystem, char?.location?.solar_system_id?.toString());
|
mapRef.current?.command(Commands.centerSystem, char?.location?.solar_system_id?.toString());
|
||||||
}, [mapRef, char]);
|
}, [mapRef, char]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -5,6 +5,11 @@
|
|||||||
.WHClassViewContent {
|
.WHClassViewContent {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 2px;
|
gap: 2px;
|
||||||
|
|
||||||
|
&.NoOffset {
|
||||||
|
gap: 4px;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.WHClassName {
|
.WHClassName {
|
||||||
@@ -13,3 +18,12 @@
|
|||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
top: -2px;
|
top: -2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.NoOffset {
|
||||||
|
*.WHClassName {
|
||||||
|
position: relative;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: initial !important;
|
||||||
|
top: initial !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -16,26 +16,42 @@ const prepareMass = (mass: number) => {
|
|||||||
|
|
||||||
export interface WHClassViewProps {
|
export interface WHClassViewProps {
|
||||||
whClassName: string;
|
whClassName: string;
|
||||||
|
noOffset?: boolean;
|
||||||
|
useShortTitle?: boolean;
|
||||||
|
hideWhClass?: boolean;
|
||||||
|
highlightName?: boolean;
|
||||||
|
className?: string;
|
||||||
|
classNameWh?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const WHClassView = ({ whClassName }: WHClassViewProps) => {
|
export const WHClassView = ({
|
||||||
|
whClassName,
|
||||||
|
noOffset,
|
||||||
|
useShortTitle,
|
||||||
|
hideWhClass,
|
||||||
|
highlightName,
|
||||||
|
className,
|
||||||
|
classNameWh,
|
||||||
|
}: WHClassViewProps) => {
|
||||||
const {
|
const {
|
||||||
data: { wormholesData },
|
data: { wormholesData },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const whData = useMemo(() => wormholesData[whClassName], [whClassName, wormholesData]);
|
const whData = useMemo(() => wormholesData[whClassName], [whClassName, wormholesData]);
|
||||||
const whClass = useMemo(() => WORMHOLES_ADDITIONAL_INFO[whData.dest], [whData.dest]);
|
const whClass = useMemo(() => WORMHOLES_ADDITIONAL_INFO[whData.dest], [whData.dest]);
|
||||||
const whClassStyle = WORMHOLE_CLASS_STYLES[whClass.wormholeClassID];
|
const whClassStyle = WORMHOLE_CLASS_STYLES[whClass?.wormholeClassID] ?? '';
|
||||||
|
|
||||||
|
const uid = useMemo(() => new Date().getTime().toString(), []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={classes.WHClassViewRoot}>
|
<div className={clsx(classes.WHClassViewRoot, className)}>
|
||||||
<Tooltip
|
<Tooltip
|
||||||
target={`.wh-name${whClassName}`}
|
target={`.wh-name${whClassName}${uid}`}
|
||||||
position="right"
|
position="right"
|
||||||
mouseTrack
|
mouseTrack
|
||||||
mouseTrackLeft={20}
|
mouseTrackLeft={20}
|
||||||
mouseTrackTop={30}
|
mouseTrackTop={30}
|
||||||
className="border border-green-300 rounded border-opacity-10 bg-stone-900 bg-opacity-70 "
|
className="border border-green-300 rounded border-opacity-10 bg-stone-900 bg-opacity-90 "
|
||||||
>
|
>
|
||||||
<div className="flex gap-3">
|
<div className="flex gap-3">
|
||||||
<div className="flex flex-col gap-1">
|
<div className="flex flex-col gap-1">
|
||||||
@@ -49,9 +65,20 @@ export const WHClassView = ({ whClassName }: WHClassViewProps) => {
|
|||||||
</div>
|
</div>
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
|
|
||||||
<div className={clsx(classes.WHClassViewContent, 'wh-name select-none cursor-help', `wh-name${whClassName}`)}>
|
<div
|
||||||
<span>{whClassName}</span>
|
className={clsx(
|
||||||
<span className={clsx(classes.WHClassName, whClassStyle)}>{whClass.shortName}</span>
|
classes.WHClassViewContent,
|
||||||
|
{ [classes.NoOffset]: noOffset },
|
||||||
|
'wh-name select-none cursor-help',
|
||||||
|
`wh-name${whClassName}${uid}`,
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<span className={clsx({ [whClassStyle]: highlightName })}>{whClassName}</span>
|
||||||
|
{!hideWhClass && whClass && (
|
||||||
|
<span className={clsx(classes.WHClassName, whClassStyle, classNameWh)}>
|
||||||
|
{useShortTitle ? whClass.shortTitle : whClass.shortName}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -5,3 +5,62 @@ export enum SESSION_KEY {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const GRADIENT_MENU_ACTIVE_CLASSES = 'bg-gradient-to-br from-transparent/10 to-fuchsia-300/10';
|
export const GRADIENT_MENU_ACTIVE_CLASSES = 'bg-gradient-to-br from-transparent/10 to-fuchsia-300/10';
|
||||||
|
|
||||||
|
export enum Regions {
|
||||||
|
Derelik = 10000001,
|
||||||
|
TheForge = 10000002,
|
||||||
|
Lonetrek = 10000016,
|
||||||
|
SinqLaison = 10000032,
|
||||||
|
Aridia = 10000054,
|
||||||
|
BlackRise = 10000069,
|
||||||
|
TheBleakLands = 10000038,
|
||||||
|
TheCitadel = 10000033,
|
||||||
|
Devoid = 10000036,
|
||||||
|
Domain = 10000043,
|
||||||
|
Essence = 10000064,
|
||||||
|
Everyshore = 10000037,
|
||||||
|
Genesis = 10000067,
|
||||||
|
Heimatar = 10000030,
|
||||||
|
Kador = 10000052,
|
||||||
|
Khanid = 10000049,
|
||||||
|
KorAzor = 10000065,
|
||||||
|
Metropolis = 10000042,
|
||||||
|
MoldenHeath = 10000028,
|
||||||
|
Placid = 10000048,
|
||||||
|
Solitude = 10000044,
|
||||||
|
TashMurkon = 10000020,
|
||||||
|
VergeVendor = 10000068,
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum Spaces {
|
||||||
|
'Caldari' = 'Caldari',
|
||||||
|
'Gallente' = 'Gallente',
|
||||||
|
'Matar' = 'Matar',
|
||||||
|
'Amarr' = 'Amarr',
|
||||||
|
}
|
||||||
|
|
||||||
|
export const REGIONS_MAP: Record<number, Spaces> = {
|
||||||
|
[Regions.Derelik]: Spaces.Amarr,
|
||||||
|
[Regions.TheForge]: Spaces.Caldari,
|
||||||
|
[Regions.Lonetrek]: Spaces.Caldari,
|
||||||
|
[Regions.SinqLaison]: Spaces.Gallente,
|
||||||
|
[Regions.Aridia]: Spaces.Amarr,
|
||||||
|
[Regions.BlackRise]: Spaces.Caldari,
|
||||||
|
[Regions.TheBleakLands]: Spaces.Amarr,
|
||||||
|
[Regions.TheCitadel]: Spaces.Caldari,
|
||||||
|
[Regions.Devoid]: Spaces.Amarr,
|
||||||
|
[Regions.Domain]: Spaces.Amarr,
|
||||||
|
[Regions.Essence]: Spaces.Gallente,
|
||||||
|
[Regions.Everyshore]: Spaces.Gallente,
|
||||||
|
[Regions.Genesis]: Spaces.Amarr,
|
||||||
|
[Regions.Heimatar]: Spaces.Matar,
|
||||||
|
[Regions.Kador]: Spaces.Amarr,
|
||||||
|
[Regions.Khanid]: Spaces.Amarr,
|
||||||
|
[Regions.KorAzor]: Spaces.Amarr,
|
||||||
|
[Regions.Metropolis]: Spaces.Matar,
|
||||||
|
[Regions.MoldenHeath]: Spaces.Matar,
|
||||||
|
[Regions.Placid]: Spaces.Gallente,
|
||||||
|
[Regions.Solitude]: Spaces.Gallente,
|
||||||
|
[Regions.TashMurkon]: Spaces.Amarr,
|
||||||
|
[Regions.VergeVendor]: Spaces.Gallente,
|
||||||
|
};
|
||||||
|
|||||||
12
assets/js/hooks/Mapper/events/index.ts
Normal file
12
assets/js/hooks/Mapper/events/index.ts
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
import { createEvent } from 'react-event-hook';
|
||||||
|
|
||||||
|
import { Command, CommandData } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
|
||||||
|
export interface MapEvent<T extends Command> {
|
||||||
|
name: T;
|
||||||
|
data: CommandData[T];
|
||||||
|
}
|
||||||
|
|
||||||
|
const { useMapEventListener, emitMapEvent } = createEvent('map-event')<MapEvent<Command>>();
|
||||||
|
|
||||||
|
export { useMapEventListener, emitMapEvent };
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { COSMIC_SIGNATURE } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures';
|
import { COSMIC_SIGNATURE } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SystemSignatureSettingsDialog';
|
||||||
import { SystemSignature } from '@/hooks/Mapper/types';
|
import { SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
export const parseSignatures = (value: string, availableKeys: string[]): SystemSignature[] => {
|
export const parseSignatures = (value: string, availableKeys: string[]): SystemSignature[] => {
|
||||||
@@ -19,6 +19,7 @@ export const parseSignatures = (value: string, availableKeys: string[]): SystemS
|
|||||||
kind: availableKeys.includes(sigArrInfo[1]) ? sigArrInfo[1] : COSMIC_SIGNATURE,
|
kind: availableKeys.includes(sigArrInfo[1]) ? sigArrInfo[1] : COSMIC_SIGNATURE,
|
||||||
group: sigArrInfo[2],
|
group: sigArrInfo[2],
|
||||||
name: sigArrInfo[3],
|
name: sigArrInfo[3],
|
||||||
|
type: '',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,10 @@ import { WORMHOLES_ADDITIONAL_INFO } from '@/hooks/Mapper/components/map/constan
|
|||||||
import { WormholeDataRaw } from '@/hooks/Mapper/types';
|
import { WormholeDataRaw } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
export const sortWHClasses = (wormholesData: Record<string, WormholeDataRaw>, statics: string[]) => {
|
export const sortWHClasses = (wormholesData: Record<string, WormholeDataRaw>, statics: string[]) => {
|
||||||
|
if (!statics) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
return statics
|
return statics
|
||||||
.map(x => wormholesData[x])
|
.map(x => wormholesData[x])
|
||||||
.map(x => ({ name: x.name, ...WORMHOLES_ADDITIONAL_INFO[x.dest] }))
|
.map(x => ({ name: x.name, ...WORMHOLES_ADDITIONAL_INFO[x.dest] }))
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
export * from './usePageVisibility';
|
export * from './usePageVisibility';
|
||||||
export * from './useClipboard';
|
export * from './useClipboard';
|
||||||
export * from './useHotkey';
|
export * from './useHotkey';
|
||||||
|
export * from './useSkipContextMenu';
|
||||||
|
|||||||
15
assets/js/hooks/Mapper/hooks/useSkipContextMenu.ts
Normal file
15
assets/js/hooks/Mapper/hooks/useSkipContextMenu.ts
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import { useEffect } from 'react';
|
||||||
|
|
||||||
|
export const useSkipContextMenu = () => {
|
||||||
|
useEffect(() => {
|
||||||
|
function handleContextMenu(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
window.addEventListener(`contextmenu`, handleContextMenu);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener(`contextmenu`, handleContextMenu);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
};
|
||||||
@@ -1,6 +1,5 @@
|
|||||||
import { createRoot } from 'react-dom/client';
|
import { createRoot } from 'react-dom/client';
|
||||||
import Mapper from './MapRoot';
|
import Mapper from './MapRoot';
|
||||||
import { decompressToJson } from './utils';
|
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
_rootEl: null,
|
_rootEl: null,
|
||||||
@@ -23,22 +22,17 @@ export default {
|
|||||||
onError: handleError,
|
onError: handleError,
|
||||||
});
|
});
|
||||||
|
|
||||||
this.pushEvent('loaded');
|
this.pushEvent('ui_loaded');
|
||||||
},
|
},
|
||||||
|
|
||||||
handleEventWrapper(event: string, handler: (payload: any) => void) {
|
handleEventWrapper(event: string, handler: (payload: any) => void) {
|
||||||
this.handleEvent(event, (body: any) => {
|
this.handleEvent(event, (body: any) => {
|
||||||
if (event === 'map_event') {
|
handler(body);
|
||||||
const { type, body: data } = body;
|
|
||||||
handler({ type, body: decompressToJson(data) });
|
|
||||||
} else {
|
|
||||||
handler(body);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
reconnected() {
|
reconnected() {
|
||||||
this.pushEvent('reconnected');
|
this.pushEvent('ui_loaded');
|
||||||
},
|
},
|
||||||
|
|
||||||
async pushEventAsync(event: string, payload: any) {
|
async pushEventAsync(event: string, payload: any) {
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ import { MapHandlers, MapUnionTypes, OutCommandHandler, SolarSystemConnection }
|
|||||||
import { useMapRootHandlers } from '@/hooks/Mapper/mapRootProvider/hooks';
|
import { useMapRootHandlers } from '@/hooks/Mapper/mapRootProvider/hooks';
|
||||||
import { WithChildren } from '@/hooks/Mapper/types/common.ts';
|
import { WithChildren } from '@/hooks/Mapper/types/common.ts';
|
||||||
import useLocalStorageState from 'use-local-storage-state';
|
import useLocalStorageState from 'use-local-storage-state';
|
||||||
import { DEFAULT_SETTINGS } from '@/hooks/Mapper/components/mapInterface/widgets/RoutesWidget/RoutesProvider.tsx';
|
|
||||||
|
|
||||||
export type MapRootData = MapUnionTypes & {
|
export type MapRootData = MapUnionTypes & {
|
||||||
selectedSystems: string[];
|
selectedSystems: string[];
|
||||||
@@ -13,6 +12,7 @@ export type MapRootData = MapUnionTypes & {
|
|||||||
|
|
||||||
const INITIAL_DATA: MapRootData = {
|
const INITIAL_DATA: MapRootData = {
|
||||||
wormholesData: {},
|
wormholesData: {},
|
||||||
|
wormholes: [],
|
||||||
effects: {},
|
effects: {},
|
||||||
characters: [],
|
characters: [],
|
||||||
userCharacters: [],
|
userCharacters: [],
|
||||||
@@ -27,14 +27,22 @@ const INITIAL_DATA: MapRootData = {
|
|||||||
selectedConnections: [],
|
selectedConnections: [],
|
||||||
};
|
};
|
||||||
|
|
||||||
type InterfaceStoredSettings = {
|
export enum InterfaceStoredSettingsProps {
|
||||||
|
isShowMenu = 'isShowMenu',
|
||||||
|
isShowMinimap = 'isShowMinimap',
|
||||||
|
isShowKSpace = 'isShowKSpace',
|
||||||
|
}
|
||||||
|
|
||||||
|
export type InterfaceStoredSettings = {
|
||||||
isShowMenu: boolean;
|
isShowMenu: boolean;
|
||||||
isShowMinimap: boolean;
|
isShowMinimap: boolean;
|
||||||
|
isShowKSpace: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const STORED_INTERFACE_DEFAULT_VALUES: InterfaceStoredSettings = {
|
export const STORED_INTERFACE_DEFAULT_VALUES: InterfaceStoredSettings = {
|
||||||
isShowMenu: false,
|
isShowMenu: false,
|
||||||
isShowMinimap: true,
|
isShowMinimap: true,
|
||||||
|
isShowKSpace: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export interface MapRootContextProps {
|
export interface MapRootContextProps {
|
||||||
@@ -50,6 +58,7 @@ const MapRootContext = createContext<MapRootContextProps>({
|
|||||||
update: () => {},
|
update: () => {},
|
||||||
data: { ...INITIAL_DATA },
|
data: { ...INITIAL_DATA },
|
||||||
mapRef: { current: null },
|
mapRef: { current: null },
|
||||||
|
// @ts-ignore
|
||||||
outCommand: async () => void 0,
|
outCommand: async () => void 0,
|
||||||
interfaceSettings: STORED_INTERFACE_DEFAULT_VALUES,
|
interfaceSettings: STORED_INTERFACE_DEFAULT_VALUES,
|
||||||
setInterfaceSettings: () => null,
|
setInterfaceSettings: () => null,
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ export const useMapInit = () => {
|
|||||||
|
|
||||||
if (wormholes) {
|
if (wormholes) {
|
||||||
updateData.wormholesData = wormholes.reduce((acc, x) => ({ ...acc, [x.name]: x }), {});
|
updateData.wormholesData = wormholes.reduce((acc, x) => ({ ...acc, [x.name]: x }), {});
|
||||||
|
updateData.wormholes = wormholes;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (effects) {
|
if (effects) {
|
||||||
|
|||||||
@@ -27,12 +27,14 @@ interface UseLoadSystemStaticProps {
|
|||||||
export const useLoadSystemStatic = ({ systems }: UseLoadSystemStaticProps) => {
|
export const useLoadSystemStatic = ({ systems }: UseLoadSystemStaticProps) => {
|
||||||
const { outCommand } = useMapRootState();
|
const { outCommand } = useMapRootState();
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
|
const [lastUpdateKey, setLastUpdateKey] = useState(0);
|
||||||
|
|
||||||
const ref = useRef({ outCommand });
|
const ref = useRef({ outCommand });
|
||||||
ref.current = { outCommand };
|
ref.current = { outCommand };
|
||||||
|
|
||||||
const addSystemStatic = useCallback((static_info: SolarSystemStaticInfoRaw) => {
|
const addSystemStatic = useCallback((static_info: SolarSystemStaticInfoRaw) => {
|
||||||
cache.set(static_info.solar_system_id, static_info);
|
cache.set(static_info.solar_system_id, static_info);
|
||||||
|
setLastUpdateKey(new Date().getTime());
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const loadSystems = useCallback(async (systems: (number | string)[]) => {
|
const loadSystems = useCallback(async (systems: (number | string)[]) => {
|
||||||
@@ -43,6 +45,7 @@ export const useLoadSystemStatic = ({ systems }: UseLoadSystemStaticProps) => {
|
|||||||
if (toLoad.length > 0) {
|
if (toLoad.length > 0) {
|
||||||
const res = await loadSystemStaticInfo(ref.current.outCommand, toLoad);
|
const res = await loadSystemStaticInfo(ref.current.outCommand, toLoad);
|
||||||
res.forEach(x => cache.set(x.solar_system_id, x));
|
res.forEach(x => cache.set(x.solar_system_id, x));
|
||||||
|
setLastUpdateKey(new Date().getTime());
|
||||||
}
|
}
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}, []);
|
}, []);
|
||||||
@@ -52,5 +55,5 @@ export const useLoadSystemStatic = ({ systems }: UseLoadSystemStaticProps) => {
|
|||||||
// eslint-disable-next-line
|
// eslint-disable-next-line
|
||||||
}, [systems]);
|
}, [systems]);
|
||||||
|
|
||||||
return { addSystemStatic, systems: cache, loading, loadSystems };
|
return { addSystemStatic, systems: cache, lastUpdateKey, loading, loadSystems };
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -27,6 +27,8 @@ import {
|
|||||||
useRoutes,
|
useRoutes,
|
||||||
} from './api';
|
} from './api';
|
||||||
|
|
||||||
|
import { emitMapEvent } from '@/hooks/Mapper/events';
|
||||||
|
|
||||||
export const useMapRootHandlers = (ref: ForwardedRef<MapHandlers>) => {
|
export const useMapRootHandlers = (ref: ForwardedRef<MapHandlers>) => {
|
||||||
const mapInit = useMapInit();
|
const mapInit = useMapInit();
|
||||||
const { addSystems, removeSystems, updateSystems } = useCommandsSystems();
|
const { addSystems, removeSystems, updateSystems } = useCommandsSystems();
|
||||||
@@ -47,15 +49,25 @@ export const useMapRootHandlers = (ref: ForwardedRef<MapHandlers>) => {
|
|||||||
break;
|
break;
|
||||||
case Commands.addSystems:
|
case Commands.addSystems:
|
||||||
addSystems(data as CommandAddSystems);
|
addSystems(data as CommandAddSystems);
|
||||||
|
setTimeout(() => {
|
||||||
|
emitMapEvent({ name: Commands.addSystems, data });
|
||||||
|
}, 100);
|
||||||
break;
|
break;
|
||||||
case Commands.updateSystems:
|
case Commands.updateSystems:
|
||||||
updateSystems(data as CommandUpdateSystems);
|
updateSystems(data as CommandUpdateSystems);
|
||||||
break;
|
break;
|
||||||
case Commands.removeSystems:
|
case Commands.removeSystems:
|
||||||
removeSystems(data as CommandRemoveSystems);
|
removeSystems(data as CommandRemoveSystems);
|
||||||
|
setTimeout(() => {
|
||||||
|
emitMapEvent({ name: Commands.removeSystems, data });
|
||||||
|
}, 100);
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case Commands.addConnections:
|
case Commands.addConnections:
|
||||||
addConnections(data as CommandAddConnections);
|
addConnections(data as CommandAddConnections);
|
||||||
|
setTimeout(() => {
|
||||||
|
emitMapEvent({ name: Commands.addConnections, data });
|
||||||
|
}, 100);
|
||||||
break;
|
break;
|
||||||
case Commands.removeConnections:
|
case Commands.removeConnections:
|
||||||
removeConnections(data as CommandRemoveConnections);
|
removeConnections(data as CommandRemoveConnections);
|
||||||
@@ -85,14 +97,30 @@ export const useMapRootHandlers = (ref: ForwardedRef<MapHandlers>) => {
|
|||||||
mapRoutes(data as CommandRoutes);
|
mapRoutes(data as CommandRoutes);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case Commands.centerSystem:
|
||||||
|
// do nothing here
|
||||||
|
break;
|
||||||
|
|
||||||
case Commands.selectSystem:
|
case Commands.selectSystem:
|
||||||
// do nothing here
|
// do nothing here
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case Commands.linkSignatureToSystem:
|
||||||
|
// TODO command data type lost
|
||||||
|
// @ts-ignore
|
||||||
|
emitMapEvent({ name: Commands.linkSignatureToSystem, data });
|
||||||
|
break;
|
||||||
|
|
||||||
case Commands.killsUpdated:
|
case Commands.killsUpdated:
|
||||||
// do nothing here
|
// do nothing here
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case Commands.signaturesUpdated:
|
||||||
|
// TODO command data type lost
|
||||||
|
// @ts-ignore
|
||||||
|
emitMapEvent({ name: Commands.signaturesUpdated, data });
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
console.warn(`JOipP Interface handlers: Unknown command: ${type}`, data);
|
console.warn(`JOipP Interface handlers: Unknown command: ${type}`, data);
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -11,6 +11,10 @@ export type Passage = {
|
|||||||
character: PassageLimitedCharacterType;
|
character: PassageLimitedCharacterType;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type ConnectionInfoOutput = {
|
||||||
|
marl_eol_time: string;
|
||||||
|
};
|
||||||
|
|
||||||
export type ConnectionOutput = {
|
export type ConnectionOutput = {
|
||||||
passages: Passage[];
|
passages: Passage[];
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -21,7 +21,10 @@ export enum Commands {
|
|||||||
mapUpdated = 'map_updated',
|
mapUpdated = 'map_updated',
|
||||||
killsUpdated = 'kills_updated',
|
killsUpdated = 'kills_updated',
|
||||||
routes = 'routes',
|
routes = 'routes',
|
||||||
|
centerSystem = 'center_system',
|
||||||
selectSystem = 'select_system',
|
selectSystem = 'select_system',
|
||||||
|
linkSignatureToSystem = 'link_signature_to_system',
|
||||||
|
signaturesUpdated = 'signatures_updated',
|
||||||
}
|
}
|
||||||
|
|
||||||
export type Command =
|
export type Command =
|
||||||
@@ -40,7 +43,10 @@ export type Command =
|
|||||||
| Commands.mapUpdated
|
| Commands.mapUpdated
|
||||||
| Commands.killsUpdated
|
| Commands.killsUpdated
|
||||||
| Commands.routes
|
| Commands.routes
|
||||||
| Commands.selectSystem;
|
| Commands.selectSystem
|
||||||
|
| Commands.centerSystem
|
||||||
|
| Commands.linkSignatureToSystem
|
||||||
|
| Commands.signaturesUpdated;
|
||||||
|
|
||||||
export type CommandInit = {
|
export type CommandInit = {
|
||||||
systems: SolarSystemRawType[];
|
systems: SolarSystemRawType[];
|
||||||
@@ -72,6 +78,12 @@ export type CommandMapUpdated = Partial<CommandInit>;
|
|||||||
export type CommandRoutes = RoutesList;
|
export type CommandRoutes = RoutesList;
|
||||||
export type CommandKillsUpdated = Kill[];
|
export type CommandKillsUpdated = Kill[];
|
||||||
export type CommandSelectSystem = string | undefined;
|
export type CommandSelectSystem = string | undefined;
|
||||||
|
export type CommandCenterSystem = string | undefined;
|
||||||
|
export type CommandLinkSignatureToSystem = {
|
||||||
|
solar_system_source: number;
|
||||||
|
solar_system_target: number;
|
||||||
|
};
|
||||||
|
export type CommandLinkSignaturesUpdated = number;
|
||||||
|
|
||||||
export interface CommandData {
|
export interface CommandData {
|
||||||
[Commands.init]: CommandInit;
|
[Commands.init]: CommandInit;
|
||||||
@@ -90,6 +102,9 @@ export interface CommandData {
|
|||||||
[Commands.routes]: CommandRoutes;
|
[Commands.routes]: CommandRoutes;
|
||||||
[Commands.killsUpdated]: CommandKillsUpdated;
|
[Commands.killsUpdated]: CommandKillsUpdated;
|
||||||
[Commands.selectSystem]: CommandSelectSystem;
|
[Commands.selectSystem]: CommandSelectSystem;
|
||||||
|
[Commands.centerSystem]: CommandCenterSystem;
|
||||||
|
[Commands.linkSignatureToSystem]: CommandLinkSignatureToSystem;
|
||||||
|
[Commands.signaturesUpdated]: CommandLinkSignaturesUpdated;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface MapHandlers {
|
export interface MapHandlers {
|
||||||
@@ -103,10 +118,12 @@ export enum OutCommand {
|
|||||||
getCharacterJumps = 'get_character_jumps',
|
getCharacterJumps = 'get_character_jumps',
|
||||||
getSignatures = 'get_signatures',
|
getSignatures = 'get_signatures',
|
||||||
getSystemStaticInfos = 'get_system_static_infos',
|
getSystemStaticInfos = 'get_system_static_infos',
|
||||||
|
getConnectionInfo = 'get_connection_info',
|
||||||
updateConnectionTimeStatus = 'update_connection_time_status',
|
updateConnectionTimeStatus = 'update_connection_time_status',
|
||||||
updateConnectionMassStatus = 'update_connection_mass_status',
|
updateConnectionMassStatus = 'update_connection_mass_status',
|
||||||
updateConnectionShipSizeType = 'update_connection_ship_size_type',
|
updateConnectionShipSizeType = 'update_connection_ship_size_type',
|
||||||
updateConnectionLocked = 'update_connection_locked',
|
updateConnectionLocked = 'update_connection_locked',
|
||||||
|
updateConnectionCustomInfo = 'update_connection_custom_info',
|
||||||
updateSignatures = 'update_signatures',
|
updateSignatures = 'update_signatures',
|
||||||
updateSystemName = 'update_system_name',
|
updateSystemName = 'update_system_name',
|
||||||
updateSystemDescription = 'update_system_description',
|
updateSystemDescription = 'update_system_description',
|
||||||
@@ -123,10 +140,16 @@ export enum OutCommand {
|
|||||||
setAutopilotWaypoint = 'set_autopilot_waypoint',
|
setAutopilotWaypoint = 'set_autopilot_waypoint',
|
||||||
addSystem = 'add_system',
|
addSystem = 'add_system',
|
||||||
addCharacter = 'add_character',
|
addCharacter = 'add_character',
|
||||||
|
openUserSettings = 'open_user_settings',
|
||||||
getPassages = 'get_passages',
|
getPassages = 'get_passages',
|
||||||
|
linkSignatureToSystem = 'link_signature_to_system',
|
||||||
|
|
||||||
// Only UI commands
|
// Only UI commands
|
||||||
openSettings = 'open_settings',
|
openSettings = 'open_settings',
|
||||||
|
|
||||||
|
getUserSettings = 'get_user_settings',
|
||||||
|
updateUserSettings = 'update_user_settings',
|
||||||
|
unlinkSignature = 'unlink_signature',
|
||||||
}
|
}
|
||||||
|
|
||||||
export type OutCommandHandler = <T = any>(event: { type: OutCommand; data: any }) => Promise<T>;
|
export type OutCommandHandler = <T = any>(event: { type: OutCommand; data: any }) => Promise<T>;
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user