mirror of
https://github.com/wanderer-industries/wanderer
synced 2025-12-03 22:35:41 +00:00
Compare commits
149 Commits
revert-94-
...
v1.44.0
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
cd0b4b0fc9 | ||
|
|
e7b115e6e6 | ||
|
|
dff8fc6396 | ||
|
|
afdaeb3d34 | ||
|
|
ac6053361e | ||
|
|
eb3e1ba3aa | ||
|
|
8468a9b5de | ||
|
|
5eafe59dcb | ||
|
|
b38bcaa8cf | ||
|
|
8a238a447d | ||
|
|
3731219216 | ||
|
|
73d5fd5f67 | ||
|
|
e8e4aed6d5 | ||
|
|
63571a462f | ||
|
|
606add4142 | ||
|
|
dac480b059 | ||
|
|
5f67cb1dd7 | ||
|
|
5886fff753 | ||
|
|
da2e12bdd1 | ||
|
|
05c3d20e56 | ||
|
|
4633d26517 | ||
|
|
30b0556d47 | ||
|
|
e094378dc5 | ||
|
|
0c48189503 | ||
|
|
a5c346627a | ||
|
|
4e526040bf | ||
|
|
869c25cd60 | ||
|
|
6aac698cd8 | ||
|
|
230016b90f | ||
|
|
4b1aef8dd9 | ||
|
|
d34509d7a0 | ||
|
|
fca98ec232 | ||
|
|
e2814e95bd | ||
|
|
68a3f84704 | ||
|
|
4bc76feefc | ||
|
|
da39a55fd0 | ||
|
|
ee3cf04cd4 | ||
|
|
d79e7fe2ff | ||
|
|
8de9fdef32 | ||
|
|
f51deeec2d | ||
|
|
a971c69a96 | ||
|
|
b7995f50de | ||
|
|
14997a2959 | ||
|
|
8fef6bcf82 | ||
|
|
1f82d23963 | ||
|
|
28317a2431 | ||
|
|
6aac496a57 | ||
|
|
ac9306b713 | ||
|
|
d55e804efa | ||
|
|
08407a5679 | ||
|
|
c37d175bec | ||
|
|
69c5326e72 | ||
|
|
305f63e11d | ||
|
|
698fd5e083 | ||
|
|
1af8342d30 | ||
|
|
68b59da78e | ||
|
|
e784a3f850 | ||
|
|
a45e2f3fc2 | ||
|
|
8a3d920c31 | ||
|
|
996d7c47bd | ||
|
|
8d2b9db430 | ||
|
|
423ce343c7 | ||
|
|
1c17912d9f | ||
|
|
6714eb5d9b | ||
|
|
1620e1fd21 | ||
|
|
859014874f | ||
|
|
ef44881f06 | ||
|
|
b0532325fa | ||
|
|
2c00bd426e | ||
|
|
6eccf2ac67 | ||
|
|
973a1e54b3 | ||
|
|
2b42b637df | ||
|
|
b950572818 | ||
|
|
e470a210f1 | ||
|
|
71ec2d413c | ||
|
|
9122412558 | ||
|
|
0ba5c963b4 | ||
|
|
39a0ce284f | ||
|
|
f9d580dbc0 | ||
|
|
5c41574328 | ||
|
|
f17d74c8b7 | ||
|
|
c88854c54c | ||
|
|
f3779961d6 | ||
|
|
d93fc29734 | ||
|
|
c67918aca5 | ||
|
|
a9f276c95a | ||
|
|
7cee4894a5 | ||
|
|
edf8bef813 | ||
|
|
2081218398 | ||
|
|
b100052453 | ||
|
|
71636e895e | ||
|
|
7ff9689b76 | ||
|
|
5a4d819622 | ||
|
|
3117d85648 | ||
|
|
114133ecd2 | ||
|
|
bf8a1197e4 | ||
|
|
54c06a1fc0 | ||
|
|
e77a42dfda | ||
|
|
f83b4a2ba7 | ||
|
|
d34e7b8d8a | ||
|
|
fa0c7f3c66 | ||
|
|
5f58645b41 | ||
|
|
7ae0ec7573 | ||
|
|
b1149cecaf | ||
|
|
8f28d2be65 | ||
|
|
d758b54ef8 | ||
|
|
58293b4dc4 | ||
|
|
f2083f4256 | ||
|
|
6c7bd5804e | ||
|
|
483ae21e89 | ||
|
|
fc36d51e24 | ||
|
|
f734565844 | ||
|
|
8c718ba181 | ||
|
|
8aaa2e7add | ||
|
|
c8d8734601 | ||
|
|
5c757e8255 | ||
|
|
22f608f302 | ||
|
|
82f90ef759 | ||
|
|
167c8eea6b | ||
|
|
d76079d4c7 | ||
|
|
bf9c4cda02 | ||
|
|
af00402546 | ||
|
|
a245842ca4 | ||
|
|
8ddd672f13 | ||
|
|
92f471c0b0 | ||
|
|
9e2a2c5b44 | ||
|
|
5f5d3df003 | ||
|
|
c66cc8868e | ||
|
|
0d6528ce4f | ||
|
|
34c385ac5f | ||
|
|
b6d12e73a9 | ||
|
|
1118858120 | ||
|
|
ae3a34d5bf | ||
|
|
43df42e49b | ||
|
|
e670f3bf03 | ||
|
|
c26a9404c5 | ||
|
|
c0fad4ca92 | ||
|
|
16dbf9378b | ||
|
|
4001fe5eac | ||
|
|
2992dd8f8b | ||
|
|
98a03d1e59 | ||
|
|
2088393c79 | ||
|
|
093042b88a | ||
|
|
e5ef35c186 | ||
|
|
1cd23d5efd | ||
|
|
ead5818a3f | ||
|
|
a5f66ada68 | ||
|
|
0919742853 | ||
|
|
9727405194 |
@@ -6,3 +6,5 @@ export EVE_CLIENT_WITH_WALLET_ID="<EVE_CLIENT_WITH_WALLET_ID>"
|
|||||||
export EVE_CLIENT_WITH_WALLET_SECRET="<EVE_CLIENT_WITH_WALLET_SECRET>"
|
export EVE_CLIENT_WITH_WALLET_SECRET="<EVE_CLIENT_WITH_WALLET_SECRET>"
|
||||||
export GIT_SHA="1111"
|
export GIT_SHA="1111"
|
||||||
export WANDERER_INVITES="false"
|
export WANDERER_INVITES="false"
|
||||||
|
export WANDERER_PUBLIC_API_DISABLED="false"
|
||||||
|
export WANDERER_ZKILL_PRELOAD_DISABLED="false"
|
||||||
|
|||||||
100
.github/workflows/build.yml
vendored
100
.github/workflows/build.yml
vendored
@@ -78,22 +78,23 @@ jobs:
|
|||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: 😅 Cache deps
|
- name: 😅 Cache deps
|
||||||
id: cache-deps
|
id: cache-deps
|
||||||
uses: actions/cache@v3
|
uses: actions/cache@v4
|
||||||
env:
|
env:
|
||||||
cache-name: cache-elixir-deps
|
cache-name: cache-elixir-deps
|
||||||
with:
|
with:
|
||||||
path: deps
|
path: |
|
||||||
key: ${{ runner.os }}-mix-${{ env.cache-name }}-${{ hashFiles('**/mix.lock') }}
|
deps
|
||||||
|
key: ${{ runner.os }}-mix-${{ matrix.elixir }}-${{ matrix.otp }}-${{ hashFiles('**/mix.lock') }}
|
||||||
restore-keys: |
|
restore-keys: |
|
||||||
${{ runner.os }}-mix-${{ env.cache-name }}-
|
${{ runner.os }}-mix-${{ matrix.elixir }}-${{ matrix.otp }}-
|
||||||
- name: 😅 Cache compiled build
|
- name: 😅 Cache compiled build
|
||||||
id: cache-build
|
id: cache-build
|
||||||
uses: actions/cache@v3
|
uses: actions/cache@v4
|
||||||
env:
|
env:
|
||||||
cache-name: cache-compiled-build
|
cache-name: cache-compiled-build
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
**/_build
|
_build
|
||||||
key: ${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-${{ hashFiles( '**/lib/**/*.{ex,eex}', '**/config/*.exs', '**/mix.exs' ) }}
|
key: ${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-${{ hashFiles( '**/lib/**/*.{ex,eex}', '**/config/*.exs', '**/mix.exs' ) }}
|
||||||
restore-keys: |
|
restore-keys: |
|
||||||
${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-
|
${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-
|
||||||
@@ -122,6 +123,9 @@ jobs:
|
|||||||
name: 🛠 Build Docker Images
|
name: 🛠 Build Docker Images
|
||||||
needs: build
|
needs: build
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
|
outputs:
|
||||||
|
release-tag: ${{ steps.get-latest-tag.outputs.tag }}
|
||||||
|
release-notes: ${{ steps.get-content.outputs.string }}
|
||||||
permissions:
|
permissions:
|
||||||
checks: write
|
checks: write
|
||||||
contents: write
|
contents: write
|
||||||
@@ -135,6 +139,7 @@ jobs:
|
|||||||
matrix:
|
matrix:
|
||||||
platform:
|
platform:
|
||||||
- linux/amd64
|
- linux/amd64
|
||||||
|
- linux/arm64
|
||||||
steps:
|
steps:
|
||||||
- name: Prepare
|
- name: Prepare
|
||||||
run: |
|
run: |
|
||||||
@@ -183,15 +188,28 @@ jobs:
|
|||||||
push: true
|
push: true
|
||||||
context: .
|
context: .
|
||||||
file: ./Dockerfile
|
file: ./Dockerfile
|
||||||
tags: ${{ env.REGISTRY_IMAGE }}:latest,${{ env.REGISTRY_IMAGE }}:${{ steps.get-latest-tag.outputs.tag }}
|
cache-from: type=gha
|
||||||
|
cache-to: type=gha,mode=max
|
||||||
labels: ${{ steps.meta.outputs.labels }}
|
labels: ${{ steps.meta.outputs.labels }}
|
||||||
platforms: ${{ matrix.platform }}
|
platforms: ${{ matrix.platform }}
|
||||||
|
outputs: type=image,"name=${{ env.REGISTRY_IMAGE }}",push-by-digest=true,name-canonical=true,push=true
|
||||||
build-args: |
|
build-args: |
|
||||||
MIX_ENV=prod
|
MIX_ENV=prod
|
||||||
BUILD_METADATA=${{ steps.meta.outputs.json }}
|
BUILD_METADATA=${{ steps.meta.outputs.json }}
|
||||||
|
|
||||||
- name: Image digest
|
- name: Export digest
|
||||||
run: echo ${{ steps.build.outputs.digest }}
|
run: |
|
||||||
|
mkdir -p /tmp/digests
|
||||||
|
digest="${{ steps.build.outputs.digest }}"
|
||||||
|
touch "/tmp/digests/${digest#sha256:}"
|
||||||
|
|
||||||
|
- name: Upload digest
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: digests-${{ env.PLATFORM_PAIR }}
|
||||||
|
path: /tmp/digests/*
|
||||||
|
if-no-files-found: error
|
||||||
|
retention-days: 1
|
||||||
|
|
||||||
- uses: markpatterson27/markdown-to-output@v1
|
- uses: markpatterson27/markdown-to-output@v1
|
||||||
id: extract-changelog
|
id: extract-changelog
|
||||||
@@ -211,16 +229,54 @@ jobs:
|
|||||||
maxLength: 500
|
maxLength: 500
|
||||||
truncationSymbol: "…"
|
truncationSymbol: "…"
|
||||||
|
|
||||||
- name: Discord Webhook Action
|
merge:
|
||||||
uses: tsickert/discord-webhook@v5.3.0
|
runs-on: ubuntu-latest
|
||||||
|
needs:
|
||||||
|
- docker
|
||||||
|
steps:
|
||||||
|
- name: Download digests
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
with:
|
with:
|
||||||
webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
|
path: /tmp/digests
|
||||||
content: ${{ steps.get-content.outputs.string }}
|
pattern: digests-*
|
||||||
|
merge-multiple: true
|
||||||
|
|
||||||
|
- name: Login to Docker Hub
|
||||||
|
uses: docker/login-action@v3
|
||||||
|
with:
|
||||||
|
username: ${{ secrets.WANDERER_DOCKER_USER }}
|
||||||
|
password: ${{ secrets.WANDERER_DOCKER_PASSWORD }}
|
||||||
|
|
||||||
|
- name: Set up Docker Buildx
|
||||||
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
|
- name: Docker meta
|
||||||
|
id: meta
|
||||||
|
uses: docker/metadata-action@v5
|
||||||
|
with:
|
||||||
|
images: |
|
||||||
|
${{ env.REGISTRY_IMAGE }}
|
||||||
|
tags: |
|
||||||
|
type=ref,event=branch
|
||||||
|
type=ref,event=pr
|
||||||
|
type=semver,pattern={{version}}
|
||||||
|
type=semver,pattern={{major}}.{{minor}}
|
||||||
|
type=semver,pattern={{version}},value=${{ needs.docker.outputs.release-tag }}
|
||||||
|
|
||||||
|
- name: Create manifest list and push
|
||||||
|
working-directory: /tmp/digests
|
||||||
|
run: |
|
||||||
|
docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \
|
||||||
|
$(printf '${{ env.REGISTRY_IMAGE }}@sha256:%s ' *)
|
||||||
|
|
||||||
|
- name: Inspect image
|
||||||
|
run: |
|
||||||
|
docker buildx imagetools inspect ${{ env.REGISTRY_IMAGE }}:${{ steps.meta.outputs.version }}
|
||||||
|
|
||||||
create-release:
|
create-release:
|
||||||
name: 🏷 Create Release
|
name: 🏷 Create Release
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
needs: docker
|
needs: [docker, merge]
|
||||||
if: ${{ github.ref == 'refs/heads/main' && github.event_name == 'push' }}
|
if: ${{ github.ref == 'refs/heads/main' && github.event_name == 'push' }}
|
||||||
steps:
|
steps:
|
||||||
- name: ⬇️ Checkout repo
|
- name: ⬇️ Checkout repo
|
||||||
@@ -228,17 +284,11 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Get Release Tag
|
|
||||||
id: get-latest-tag
|
|
||||||
uses: "WyriHaximus/github-action-get-previous-tag@v1"
|
|
||||||
with:
|
|
||||||
fallback: 1.0.0
|
|
||||||
|
|
||||||
- name: 🏷 Create Draft Release
|
- name: 🏷 Create Draft Release
|
||||||
uses: softprops/action-gh-release@v1
|
uses: softprops/action-gh-release@v1
|
||||||
with:
|
with:
|
||||||
tag_name: ${{ steps.get-latest-tag.outputs.tag }}
|
tag_name: ${{ needs.docker.outputs.release-tag }}
|
||||||
name: Release ${{ steps.get-latest-tag.outputs.tag }}
|
name: Release ${{ needs.docker.outputs.release-tag }}
|
||||||
body: |
|
body: |
|
||||||
## Info
|
## Info
|
||||||
Commit ${{ github.sha }} was deployed to `staging`. [See code diff](${{ github.event.compare }}).
|
Commit ${{ github.sha }} was deployed to `staging`. [See code diff](${{ github.event.compare }}).
|
||||||
@@ -248,3 +298,9 @@ jobs:
|
|||||||
## How to Promote?
|
## How to Promote?
|
||||||
In order to promote this to prod, edit the draft and press **"Publish release"**.
|
In order to promote this to prod, edit the draft and press **"Publish release"**.
|
||||||
draft: true
|
draft: true
|
||||||
|
|
||||||
|
- name: Discord Webhook Action
|
||||||
|
uses: tsickert/discord-webhook@v5.3.0
|
||||||
|
with:
|
||||||
|
webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
|
||||||
|
content: ${{ needs.docker.outputs.release-notes }}
|
||||||
|
|||||||
1711
CHANGELOG.md
1711
CHANGELOG.md
File diff suppressed because it is too large
Load Diff
@@ -9,6 +9,9 @@ WORKDIR /app
|
|||||||
# set build ENV
|
# set build ENV
|
||||||
ENV MIX_ENV="prod"
|
ENV MIX_ENV="prod"
|
||||||
|
|
||||||
|
# Set ERL_FLAGS for ARM compatibility
|
||||||
|
ENV ERL_FLAGS="+JPperf true"
|
||||||
|
|
||||||
# install mix dependencies
|
# install mix dependencies
|
||||||
COPY mix.exs mix.lock ./
|
COPY mix.exs mix.lock ./
|
||||||
RUN rm -Rf _build deps && mix deps.get --only $MIX_ENV
|
RUN rm -Rf _build deps && mix deps.get --only $MIX_ENV
|
||||||
|
|||||||
@@ -3,6 +3,8 @@
|
|||||||
@import 'primereact/resources/themes/arya-blue/theme.css' layer(primereact);
|
@import 'primereact/resources/themes/arya-blue/theme.css' layer(primereact);
|
||||||
/*@import 'primereact/resources/themes/bootstrap4-dark-blue/theme.css' layer(primereact);*/
|
/*@import 'primereact/resources/themes/bootstrap4-dark-blue/theme.css' layer(primereact);*/
|
||||||
|
|
||||||
|
@import '../js/hooks/Mapper/components/map/styles/index.scss';
|
||||||
|
|
||||||
@layer tailwind-base {
|
@layer tailwind-base {
|
||||||
@tailwind base;
|
@tailwind base;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -112,3 +112,28 @@
|
|||||||
.p-autocomplete .p-autocomplete-multiple-container .p-autocomplete-token {
|
.p-autocomplete .p-autocomplete-multiple-container .p-autocomplete-token {
|
||||||
margin-right: 0 !important;
|
margin-right: 0 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Fixed sizes of Input switch */
|
||||||
|
.p-inputswitch {
|
||||||
|
width: 2.0rem;
|
||||||
|
height: 1.15rem;
|
||||||
|
|
||||||
|
.p-inputswitch-slider:before {
|
||||||
|
width: 0.8rem;
|
||||||
|
height: 0.8rem;
|
||||||
|
left: 0.14rem;
|
||||||
|
margin-top: -0.385rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.p-highlight .p-inputswitch-slider:before {
|
||||||
|
transform: translateX(0.8rem);
|
||||||
|
}
|
||||||
|
|
||||||
|
&:not(.p-disabled):has(.p-inputswitch-input:hover) .p-inputswitch-slider {
|
||||||
|
background: rgb(255 255 255 / 21%);
|
||||||
|
}
|
||||||
|
|
||||||
|
&.p-highlight .p-inputswitch-slider {
|
||||||
|
background: #966d3d;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -88,6 +88,23 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
|
|||||||
setSystem(undefined);
|
setSystem(undefined);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const onSystemTemporaryName = useCallback((temporaryName?: string) => {
|
||||||
|
const { system, outCommand } = ref.current;
|
||||||
|
if (!system) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
outCommand({
|
||||||
|
type: OutCommand.updateSystemTemporaryName,
|
||||||
|
data: {
|
||||||
|
system_id: system,
|
||||||
|
value: temporaryName ?? '',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
setSystem(undefined);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
|
||||||
const onSystemStatus = useCallback((status: number) => {
|
const onSystemStatus = useCallback((status: number) => {
|
||||||
const { system, outCommand } = ref.current;
|
const { system, outCommand } = ref.current;
|
||||||
if (!system) {
|
if (!system) {
|
||||||
@@ -161,6 +178,7 @@ export const useContextMenuSystemHandlers = ({ systems, hubs, outCommand }: UseC
|
|||||||
onLockToggle,
|
onLockToggle,
|
||||||
onHubToggle,
|
onHubToggle,
|
||||||
onSystemTag,
|
onSystemTag,
|
||||||
|
onSystemTemporaryName,
|
||||||
onSystemStatus,
|
onSystemStatus,
|
||||||
onSystemLabels,
|
onSystemLabels,
|
||||||
onOpenSettings,
|
onOpenSettings,
|
||||||
|
|||||||
@@ -1,8 +1,11 @@
|
|||||||
.MapRoot {
|
.MapRoot {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
|
||||||
|
|
||||||
.BackgroundAlternateColor {
|
background-color: var(--rf-bg-color, #0C0A09);
|
||||||
|
|
||||||
|
&.BackgroundAlternateColor {
|
||||||
|
background-color: var(--rf-soft-bg-color, #171717);
|
||||||
|
--rf-node-bg-color: var(--rf-node-soft-bg-color, #202020);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { ForwardedRef, forwardRef, MouseEvent, useCallback, useEffect } from 'react';
|
import { ForwardedRef, forwardRef, MouseEvent, useCallback, useEffect, useMemo } from 'react';
|
||||||
import ReactFlow, {
|
import ReactFlow, {
|
||||||
Background,
|
Background,
|
||||||
ConnectionMode,
|
|
||||||
Edge,
|
Edge,
|
||||||
|
EdgeChange,
|
||||||
MiniMap,
|
MiniMap,
|
||||||
Node,
|
Node,
|
||||||
NodeChange,
|
NodeChange,
|
||||||
@@ -16,25 +16,24 @@ import ReactFlow, {
|
|||||||
} 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/eve-common.scss';
|
|
||||||
import { MapProvider, useMapState } from './MapProvider';
|
import { MapProvider, useMapState } from './MapProvider';
|
||||||
import { useNodesState, useEdgesState, useMapHandlers, useUpdateNodes } from './hooks';
|
import { useEdgesState, useMapHandlers, useNodesState, useUpdateNodes } from './hooks';
|
||||||
import { MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import {
|
import {
|
||||||
ContextMenuConnection,
|
ContextMenuConnection,
|
||||||
ContextMenuRoot,
|
ContextMenuRoot,
|
||||||
SolarSystemEdge,
|
SolarSystemEdge,
|
||||||
SolarSystemNode,
|
|
||||||
useContextMenuConnectionHandlers,
|
useContextMenuConnectionHandlers,
|
||||||
useContextMenuRootHandlers,
|
useContextMenuRootHandlers,
|
||||||
} from './components';
|
} from './components';
|
||||||
|
import { getBehaviorForTheme } from './helpers/getThemeBehavior';
|
||||||
import { OnMapAddSystemCallback, OnMapSelectionChange } from './map.types';
|
import { OnMapAddSystemCallback, OnMapSelectionChange } from './map.types';
|
||||||
import { SESSION_KEY } from '@/hooks/Mapper/constants.ts';
|
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 clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
|
import { useBackgroundVars } from './hooks/useBackgroundVars';
|
||||||
|
|
||||||
const DEFAULT_VIEW_PORT = { zoom: 1, x: 0, y: 0 };
|
const DEFAULT_VIEW_PORT = { zoom: 1, x: 0, y: 0 };
|
||||||
|
|
||||||
@@ -76,12 +75,6 @@ const initialEdges = [
|
|||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const nodeTypes = {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
||||||
// @ts-ignore
|
|
||||||
custom: SolarSystemNode,
|
|
||||||
} as never;
|
|
||||||
|
|
||||||
const edgeTypes = {
|
const edgeTypes = {
|
||||||
floating: SolarSystemEdge,
|
floating: SolarSystemEdge,
|
||||||
};
|
};
|
||||||
@@ -91,6 +84,7 @@ interface MapCompProps {
|
|||||||
onCommand: OutCommandHandler;
|
onCommand: OutCommandHandler;
|
||||||
onSelectionChange: OnMapSelectionChange;
|
onSelectionChange: OnMapSelectionChange;
|
||||||
onManualDelete(systems: string[]): void;
|
onManualDelete(systems: string[]): void;
|
||||||
|
canRemoveConnection?(connectionId: string): boolean;
|
||||||
onConnectionInfoClick?(e: SolarSystemConnection): void;
|
onConnectionInfoClick?(e: SolarSystemConnection): void;
|
||||||
onAddSystem?: OnMapAddSystemCallback;
|
onAddSystem?: OnMapAddSystemCallback;
|
||||||
onSelectionContextMenu?: NodeSelectionMouseHandler;
|
onSelectionContextMenu?: NodeSelectionMouseHandler;
|
||||||
@@ -101,6 +95,7 @@ interface MapCompProps {
|
|||||||
isThickConnections?: boolean;
|
isThickConnections?: boolean;
|
||||||
isShowBackgroundPattern?: boolean;
|
isShowBackgroundPattern?: boolean;
|
||||||
isSoftBackground?: boolean;
|
isSoftBackground?: boolean;
|
||||||
|
theme?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const MapComp = ({
|
const MapComp = ({
|
||||||
@@ -117,9 +112,11 @@ const MapComp = ({
|
|||||||
isThickConnections,
|
isThickConnections,
|
||||||
isShowBackgroundPattern,
|
isShowBackgroundPattern,
|
||||||
isSoftBackground,
|
isSoftBackground,
|
||||||
|
theme,
|
||||||
onAddSystem,
|
onAddSystem,
|
||||||
|
canRemoveConnection,
|
||||||
}: MapCompProps) => {
|
}: MapCompProps) => {
|
||||||
const { getNode, getNodes } = useReactFlow();
|
const { getEdge, getNode, getNodes } = useReactFlow();
|
||||||
const [nodes, , onNodesChange] = useNodesState<Node<SolarSystemRawType>>(initialNodes);
|
const [nodes, , onNodesChange] = useNodesState<Node<SolarSystemRawType>>(initialNodes);
|
||||||
const [edges, , onEdgesChange] = useEdgesState<Edge<SolarSystemConnection>>(initialEdges);
|
const [edges, , onEdgesChange] = useEdgesState<Edge<SolarSystemConnection>>(initialEdges);
|
||||||
|
|
||||||
@@ -128,6 +125,15 @@ const MapComp = ({
|
|||||||
const { handleRootContext, ...rootCtxProps } = useContextMenuRootHandlers({ onAddSystem });
|
const { handleRootContext, ...rootCtxProps } = useContextMenuRootHandlers({ onAddSystem });
|
||||||
const { handleConnectionContext, ...connectionCtxProps } = useContextMenuConnectionHandlers();
|
const { handleConnectionContext, ...connectionCtxProps } = useContextMenuConnectionHandlers();
|
||||||
const { update } = useMapState();
|
const { update } = useMapState();
|
||||||
|
const { variant, gap, size, color } = useBackgroundVars(theme);
|
||||||
|
const { isPanAndDrag, nodeComponent, connectionMode } = getBehaviorForTheme(theme || 'default');
|
||||||
|
|
||||||
|
// You can create nodeTypes dynamically based on the node component
|
||||||
|
const nodeTypes = useMemo(() => {
|
||||||
|
return {
|
||||||
|
custom: nodeComponent,
|
||||||
|
};
|
||||||
|
}, [nodeComponent]);
|
||||||
|
|
||||||
const onConnect: OnConnect = useCallback(
|
const onConnect: OnConnect = useCallback(
|
||||||
params => {
|
params => {
|
||||||
@@ -216,7 +222,41 @@ const MapComp = ({
|
|||||||
|
|
||||||
onNodesChange(nextChanges);
|
onNodesChange(nextChanges);
|
||||||
},
|
},
|
||||||
[getNode, onManualDelete, onNodesChange],
|
[getNode, getNodes, onManualDelete, onNodesChange],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleEdgesChange = useCallback(
|
||||||
|
(changes: EdgeChange[]) => {
|
||||||
|
const nextChanges = changes.reduce((acc, change) => {
|
||||||
|
if (change.type !== 'remove') {
|
||||||
|
return [...acc, change];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (canRemoveConnection?.(change.id)) {
|
||||||
|
return [...acc, change];
|
||||||
|
}
|
||||||
|
|
||||||
|
const edge = getEdge(change.id);
|
||||||
|
if (!edge) {
|
||||||
|
return [...acc, change];
|
||||||
|
}
|
||||||
|
|
||||||
|
const sourceNode = getNode(edge.source);
|
||||||
|
const targetNode = getNode(edge.target);
|
||||||
|
if (!sourceNode || !targetNode) {
|
||||||
|
return [...acc, change];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sourceNode.data.locked || targetNode.data.locked) {
|
||||||
|
return acc;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [...acc, change];
|
||||||
|
}, [] as EdgeChange[]);
|
||||||
|
|
||||||
|
onEdgesChange(nextChanges);
|
||||||
|
},
|
||||||
|
[getEdge, getNode, onEdgesChange],
|
||||||
);
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
@@ -229,19 +269,19 @@ const MapComp = ({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className={clsx(classes.MapRoot, { ['bg-neutral-900']: isSoftBackground })}>
|
<div className={clsx(classes.MapRoot, { [classes.BackgroundAlternateColor]: isSoftBackground })}>
|
||||||
<ReactFlow
|
<ReactFlow
|
||||||
nodes={nodes}
|
nodes={nodes}
|
||||||
edges={edges}
|
edges={edges}
|
||||||
onNodesChange={handleNodesChange}
|
onNodesChange={handleNodesChange}
|
||||||
onEdgesChange={onEdgesChange}
|
onEdgesChange={handleEdgesChange}
|
||||||
onConnect={onConnect}
|
onConnect={onConnect}
|
||||||
// TODO we need save into session all of this
|
// TODO we need save into session all of this
|
||||||
// and on any action do either
|
// and on any action do either
|
||||||
defaultViewport={getViewPortFromStore()}
|
defaultViewport={getViewPortFromStore()}
|
||||||
edgeTypes={edgeTypes}
|
edgeTypes={edgeTypes}
|
||||||
nodeTypes={nodeTypes}
|
nodeTypes={nodeTypes}
|
||||||
connectionMode={ConnectionMode.Loose}
|
connectionMode={connectionMode}
|
||||||
snapToGrid
|
snapToGrid
|
||||||
nodeDragThreshold={10}
|
nodeDragThreshold={10}
|
||||||
onNodeDragStop={handleDragStop}
|
onNodeDragStop={handleDragStop}
|
||||||
@@ -249,6 +289,10 @@ const MapComp = ({
|
|||||||
onConnectStart={() => update({ isConnecting: true })}
|
onConnectStart={() => update({ isConnecting: true })}
|
||||||
onConnectEnd={() => update({ isConnecting: false })}
|
onConnectEnd={() => update({ isConnecting: false })}
|
||||||
onNodeMouseEnter={(_, node) => update({ hoverNodeId: node.id })}
|
onNodeMouseEnter={(_, node) => update({ hoverNodeId: node.id })}
|
||||||
|
onPaneClick={event => {
|
||||||
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
|
}}
|
||||||
// onKeyUp=
|
// onKeyUp=
|
||||||
onNodeMouseLeave={() => update({ hoverNodeId: null })}
|
onNodeMouseLeave={() => update({ hoverNodeId: null })}
|
||||||
onEdgeClick={(_, t) => {
|
onEdgeClick={(_, t) => {
|
||||||
@@ -270,13 +314,19 @@ const MapComp = ({
|
|||||||
maxZoom={1.5}
|
maxZoom={1.5}
|
||||||
elevateNodesOnSelect
|
elevateNodesOnSelect
|
||||||
deleteKeyCode={['Delete']}
|
deleteKeyCode={['Delete']}
|
||||||
|
{...(isPanAndDrag
|
||||||
|
? {
|
||||||
|
selectionOnDrag: true,
|
||||||
|
panOnDrag: [2],
|
||||||
|
}
|
||||||
|
: {})}
|
||||||
// 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
|
||||||
selectionMode={SelectionMode.Partial}
|
selectionMode={SelectionMode.Partial}
|
||||||
>
|
>
|
||||||
{isShowMinimap && <MiniMap pannable zoomable ariaLabel="Mini map" className={minimapClasses} />}
|
{isShowMinimap && <MiniMap pannable zoomable ariaLabel="Mini map" className={minimapClasses} />}
|
||||||
{isShowBackgroundPattern && <Background />}
|
{isShowBackgroundPattern && <Background variant={variant} gap={gap} size={size} color={color} />}
|
||||||
</ReactFlow>
|
</ReactFlow>
|
||||||
{/* <button className="z-auto btn btn-primary absolute top-20 right-20" onClick={handleGetPassages}>
|
{/* <button className="z-auto btn btn-primary absolute top-20 right-20" onClick={handleGetPassages}>
|
||||||
Test // DON NOT REMOVE
|
Test // DON NOT REMOVE
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import React, { createContext, useContext } from 'react';
|
import React, { createContext, useContext } from 'react';
|
||||||
import { OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { MapUnionTypes } from '@/hooks/Mapper/types';
|
import { MapUnionTypes, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
import { ContextStoreDataUpdate, useContextStore } from '@/hooks/Mapper/utils';
|
import { ContextStoreDataUpdate, useContextStore } from '@/hooks/Mapper/utils';
|
||||||
|
|
||||||
export type MapData = MapUnionTypes & {
|
export type MapData = MapUnionTypes & {
|
||||||
@@ -9,6 +9,7 @@ export type MapData = MapUnionTypes & {
|
|||||||
visibleNodes: Set<string>;
|
visibleNodes: Set<string>;
|
||||||
showKSpaceBG: boolean;
|
showKSpaceBG: boolean;
|
||||||
isThickConnections: boolean;
|
isThickConnections: boolean;
|
||||||
|
linkedSigEveId: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
interface MapProviderProps {
|
interface MapProviderProps {
|
||||||
@@ -29,10 +30,13 @@ const INITIAL_DATA: MapData = {
|
|||||||
isConnecting: false,
|
isConnecting: false,
|
||||||
connections: [],
|
connections: [],
|
||||||
hoverNodeId: null,
|
hoverNodeId: null,
|
||||||
|
linkedSigEveId: '',
|
||||||
visibleNodes: new Set(),
|
visibleNodes: new Set(),
|
||||||
showKSpaceBG: false,
|
showKSpaceBG: false,
|
||||||
isThickConnections: false,
|
isThickConnections: false,
|
||||||
userPermissions: {},
|
userPermissions: {},
|
||||||
|
systemSignatures: {} as Record<string, SystemSignature[]>,
|
||||||
|
options: {} as Record<string, string | boolean>,
|
||||||
};
|
};
|
||||||
|
|
||||||
export interface MapContextProps {
|
export interface MapContextProps {
|
||||||
|
|||||||
@@ -1,15 +1,17 @@
|
|||||||
|
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
||||||
|
|
||||||
.ConnectionTimeEOL {
|
.ConnectionTimeEOL {
|
||||||
background-image: linear-gradient(207deg, transparent, #7452c3e3);
|
background-image: linear-gradient(207deg, transparent, var(--conn-time-eol));
|
||||||
}
|
}
|
||||||
|
|
||||||
.ConnectionFrigate {
|
.ConnectionFrigate {
|
||||||
background-image: linear-gradient(207deg, transparent, #325d88);
|
background-image: linear-gradient(207deg, transparent, var(--conn-frigate));
|
||||||
}
|
}
|
||||||
|
|
||||||
.ConnectionSave {
|
.ConnectionSave {
|
||||||
background-image: linear-gradient(207deg, transparent, rgba(155, 102, 45, 0.85));
|
background-image: linear-gradient(207deg, transparent, var(--conn-save));
|
||||||
}
|
}
|
||||||
|
|
||||||
.SelectedItem {
|
.SelectedItem {
|
||||||
background-color: rgba(98, 98, 98, 0.33);
|
background-color: var(--selected-item-bg);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,38 @@
|
|||||||
@import "@/hooks/Mapper/components/map/styles/neon-variables";
|
@import '@/hooks/Mapper/components/map/styles/eve-common-variables';
|
||||||
|
|
||||||
.react-flow__edge.selected {
|
.EdgePathBack {
|
||||||
.EdgePathBack {
|
fill: none;
|
||||||
stroke: $pastel-yellow;
|
stroke: #80a5c5;
|
||||||
|
stroke-width: 3px;
|
||||||
|
|
||||||
|
&.TimeCrit {
|
||||||
|
stroke: #f11ab2;
|
||||||
|
stroke-width: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.Hovered {
|
||||||
|
stroke: #b5c8d9;
|
||||||
|
|
||||||
|
&.TimeCrit {
|
||||||
|
stroke: #ef7dce;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.Tick {
|
||||||
|
stroke-width: 5px;
|
||||||
|
|
||||||
|
&.TimeCrit {
|
||||||
|
stroke-width: 6px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&.Gate {
|
||||||
|
stroke: #9aff40;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.EdgePathFront {
|
.EdgePathFront {
|
||||||
fill: none;
|
fill: none;
|
||||||
|
|
||||||
stroke: #2c3844;
|
stroke: #2c3844;
|
||||||
stroke-width: 2px;
|
stroke-width: 2px;
|
||||||
|
|
||||||
@@ -54,46 +78,29 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.EdgePathBack {
|
|
||||||
fill: none;
|
|
||||||
|
|
||||||
stroke: #80a5c5;
|
|
||||||
stroke-width: 3px;
|
|
||||||
|
|
||||||
&.TimeCrit {
|
|
||||||
stroke: #f11ab2;
|
|
||||||
stroke-width: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.Hovered {
|
|
||||||
stroke: #b5c8d9;
|
|
||||||
|
|
||||||
&.TimeCrit {
|
|
||||||
stroke: #ef7dce;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
&.Tick {
|
|
||||||
stroke-width: 5px;
|
|
||||||
|
|
||||||
&.TimeCrit {
|
|
||||||
stroke-width: 6px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
&.Gate {
|
|
||||||
stroke: #9aff40;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
.ClickPath {
|
.ClickPath {
|
||||||
fill: none;
|
fill: none;
|
||||||
stroke: none;
|
stroke: none;
|
||||||
stroke-width: 8px;
|
stroke-width: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.LinkLabel{
|
.Handle {
|
||||||
|
border: 1px solid var(--pastel-blue);
|
||||||
|
width: 5px;
|
||||||
|
height: 5px;
|
||||||
|
z-index: 1001;
|
||||||
|
|
||||||
|
&.Tick {
|
||||||
|
width: 7px;
|
||||||
|
height: 7px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&.Right {
|
||||||
|
margin-left: 0px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.LinkLabel {
|
||||||
font-size: 9px;
|
font-size: 9px;
|
||||||
line-height: 10px;
|
line-height: 10px;
|
||||||
padding: 2px 4px;
|
padding: 2px 4px;
|
||||||
@@ -109,22 +116,3 @@
|
|||||||
height: 8px;
|
height: 8px;
|
||||||
font-size: 8px;
|
font-size: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.Handle {
|
|
||||||
min-width: initial;
|
|
||||||
min-height: initial;
|
|
||||||
border: 1px solid #5a7d9a;
|
|
||||||
width: 5px;
|
|
||||||
height: 5px;
|
|
||||||
z-index: 1001;
|
|
||||||
|
|
||||||
&.Tick {
|
|
||||||
width: 7px;
|
|
||||||
height: 7px;
|
|
||||||
|
|
||||||
&.Right {
|
|
||||||
margin-left: 0px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|||||||
@@ -130,7 +130,7 @@ export const SolarSystemEdge = ({ id, source, target, markerEnd, style, data }:
|
|||||||
/>
|
/>
|
||||||
|
|
||||||
<div
|
<div
|
||||||
className="absolute flex items-center gap-1"
|
className="absolute flex items-center gap-1 pointer-events-none"
|
||||||
style={{
|
style={{
|
||||||
transform: `translate(-50%, -50%) translate(${labelX}px,${labelY}px)`,
|
transform: `translate(-50%, -50%) translate(${labelX}px,${labelY}px)`,
|
||||||
}}
|
}}
|
||||||
|
|||||||
@@ -1,324 +0,0 @@
|
|||||||
import { memo, useMemo } from 'react';
|
|
||||||
import { Handle, Position, WrapNodeProps } from 'reactflow';
|
|
||||||
import { MapSolarSystemType } from '../../map.types';
|
|
||||||
import classes from './SolarSystemNode.module.scss';
|
|
||||||
import clsx from 'clsx';
|
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
|
||||||
|
|
||||||
import {
|
|
||||||
EFFECT_BACKGROUND_STYLES,
|
|
||||||
LABELS_INFO,
|
|
||||||
LABELS_ORDER,
|
|
||||||
MARKER_BOOKMARK_BG_STYLES,
|
|
||||||
STATUS_CLASSES,
|
|
||||||
} from '@/hooks/Mapper/components/map/constants.ts';
|
|
||||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace.ts';
|
|
||||||
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
|
||||||
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
|
||||||
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
|
||||||
import { getSystemClassStyles, prepareUnsplashedChunks } from '@/hooks/Mapper/components/map/helpers';
|
|
||||||
import { sortWHClasses } from '@/hooks/Mapper/helpers';
|
|
||||||
import { PrimeIcons } from 'primereact/api';
|
|
||||||
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager.ts';
|
|
||||||
import { OutCommand } from '@/hooks/Mapper/types';
|
|
||||||
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick.ts';
|
|
||||||
import { REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
|
|
||||||
|
|
||||||
const SpaceToClass: Record<string, string> = {
|
|
||||||
[Spaces.Caldari]: classes.Caldaria,
|
|
||||||
[Spaces.Matar]: classes.Mataria,
|
|
||||||
[Spaces.Amarr]: classes.Amarria,
|
|
||||||
[Spaces.Gallente]: classes.Gallente,
|
|
||||||
};
|
|
||||||
|
|
||||||
const sortedLabels = (labels: string[]) => {
|
|
||||||
if (!labels) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
return LABELS_ORDER.filter(x => labels.includes(x)).map(x => LABELS_INFO[x]);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const getActivityType = (count: number) => {
|
|
||||||
if (count <= 5) {
|
|
||||||
return 'activityNormal';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (count <= 30) {
|
|
||||||
return 'activityWarn';
|
|
||||||
}
|
|
||||||
|
|
||||||
return 'activityDanger';
|
|
||||||
};
|
|
||||||
|
|
||||||
// eslint-disable-next-line react/display-name
|
|
||||||
export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarSystemType>) => {
|
|
||||||
const { interfaceSettings } = useMapRootState();
|
|
||||||
const { isShowUnsplashedSignatures } = interfaceSettings;
|
|
||||||
|
|
||||||
const {
|
|
||||||
system_class,
|
|
||||||
security,
|
|
||||||
class_title,
|
|
||||||
solar_system_id,
|
|
||||||
statics,
|
|
||||||
effect_name,
|
|
||||||
region_name,
|
|
||||||
region_id,
|
|
||||||
is_shattered,
|
|
||||||
solar_system_name,
|
|
||||||
} = data.system_static_info;
|
|
||||||
|
|
||||||
const signatures = data.system_signatures;
|
|
||||||
|
|
||||||
const { locked, name, tag, status, labels, id } = data || {};
|
|
||||||
|
|
||||||
const customName = solar_system_name !== name ? name : undefined;
|
|
||||||
|
|
||||||
const {
|
|
||||||
data: {
|
|
||||||
characters,
|
|
||||||
presentCharacters,
|
|
||||||
wormholesData,
|
|
||||||
hubs,
|
|
||||||
kills,
|
|
||||||
userCharacters,
|
|
||||||
isConnecting,
|
|
||||||
hoverNodeId,
|
|
||||||
visibleNodes,
|
|
||||||
showKSpaceBG,
|
|
||||||
isThickConnections,
|
|
||||||
},
|
|
||||||
outCommand,
|
|
||||||
} = useMapState();
|
|
||||||
|
|
||||||
const visible = useMemo(() => visibleNodes.has(id), [id, visibleNodes]);
|
|
||||||
|
|
||||||
const charactersInSystem = useMemo(() => {
|
|
||||||
return characters.filter(c => c.location?.solar_system_id === solar_system_id).filter(c => c.online);
|
|
||||||
// eslint-disable-next-line
|
|
||||||
}, [characters, presentCharacters, solar_system_id]);
|
|
||||||
|
|
||||||
const isWormhole = isWormholeSpace(system_class);
|
|
||||||
const classTitleColor = useMemo(
|
|
||||||
() => getSystemClassStyles({ systemClass: system_class, security }),
|
|
||||||
[security, system_class],
|
|
||||||
);
|
|
||||||
const sortedStatics = useMemo(() => sortWHClasses(wormholesData, statics), [wormholesData, statics]);
|
|
||||||
const lebM = useMemo(() => new LabelsManager(labels ?? ''), [labels]);
|
|
||||||
const labelsInfo = useMemo(() => sortedLabels(lebM.list), [lebM]);
|
|
||||||
const labelCustom = useMemo(() => lebM.customLabel, [lebM]);
|
|
||||||
|
|
||||||
const killsCount = useMemo(() => {
|
|
||||||
const systemKills = kills[solar_system_id];
|
|
||||||
if (!systemKills) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return systemKills;
|
|
||||||
}, [kills, solar_system_id]);
|
|
||||||
|
|
||||||
const hasUserCharacters = useMemo(() => {
|
|
||||||
return charactersInSystem.some(x => userCharacters.includes(x.eve_id));
|
|
||||||
}, [charactersInSystem, userCharacters]);
|
|
||||||
|
|
||||||
const dbClick = useDoubleClick(() => {
|
|
||||||
outCommand({
|
|
||||||
type: OutCommand.openSettings,
|
|
||||||
data: {
|
|
||||||
system_id: solar_system_id.toString(),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
const showHandlers = isConnecting || hoverNodeId === id;
|
|
||||||
|
|
||||||
const space = showKSpaceBG ? REGIONS_MAP[region_id] : '';
|
|
||||||
const regionClass = showKSpaceBG ? SpaceToClass[space] : null;
|
|
||||||
|
|
||||||
const [unsplashedLeft, unsplashedRight] = useMemo(() => {
|
|
||||||
if (!isShowUnsplashedSignatures) {
|
|
||||||
return [[], []];
|
|
||||||
}
|
|
||||||
|
|
||||||
return prepareUnsplashedChunks(
|
|
||||||
signatures
|
|
||||||
.filter(s => s.group === 'Wormhole' && !s.linked_system)
|
|
||||||
.map(s => ({
|
|
||||||
eve_id: s.eve_id,
|
|
||||||
type: s.type,
|
|
||||||
custom_info: s.custom_info,
|
|
||||||
})),
|
|
||||||
);
|
|
||||||
}, [isShowUnsplashedSignatures, signatures]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
{visible && (
|
|
||||||
<div className={classes.Bookmarks}>
|
|
||||||
{labelCustom !== '' && (
|
|
||||||
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
|
||||||
<span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] ">{labelCustom}</span>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{is_shattered && (
|
|
||||||
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.shattered)}>
|
|
||||||
<span className={clsx('pi pi-chart-pie', classes.icon)} />
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{killsCount && (
|
|
||||||
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[getActivityType(killsCount)])}>
|
|
||||||
<div className={clsx(classes.BookmarkWithIcon)}>
|
|
||||||
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
|
||||||
<span className={clsx(classes.text)}>{killsCount}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{labelsInfo.map(x => (
|
|
||||||
<div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
|
|
||||||
{x.shortName}
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<div
|
|
||||||
className={clsx(classes.RootCustomNode, regionClass, classes[STATUS_CLASSES[status]], {
|
|
||||||
[classes.selected]: selected,
|
|
||||||
})}
|
|
||||||
>
|
|
||||||
{visible && (
|
|
||||||
<>
|
|
||||||
<div className={classes.HeadRow}>
|
|
||||||
<div className={clsx(classes.classTitle, classTitleColor, '[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]')}>
|
|
||||||
{class_title ?? '-'}
|
|
||||||
</div>
|
|
||||||
{tag != null && tag !== '' && (
|
|
||||||
<div className={clsx(classes.TagTitle, 'text-sky-400 font-medium')}>{tag}</div>
|
|
||||||
)}
|
|
||||||
<div
|
|
||||||
className={clsx(
|
|
||||||
classes.classSystemName,
|
|
||||||
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] flex-grow overflow-hidden text-ellipsis whitespace-nowrap font-sans',
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{solar_system_name}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{isWormhole && (
|
|
||||||
<div className={classes.statics}>
|
|
||||||
{sortedStatics.map(x => (
|
|
||||||
<WormholeClassComp key={x} id={x} />
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{effect_name !== null && isWormhole && (
|
|
||||||
<div className={clsx(classes.effect, EFFECT_BACKGROUND_STYLES[effect_name])}></div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
|
|
||||||
{customName && (
|
|
||||||
<div className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] text-blue-300 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
|
|
||||||
{customName}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{!isWormhole && !customName && (
|
|
||||||
<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}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{isWormhole && !customName && <div />}
|
|
||||||
|
|
||||||
<div className="flex items-center justify-end">
|
|
||||||
<div className="flex gap-1 items-center">
|
|
||||||
{locked && <i className={PrimeIcons.LOCK} style={{ fontSize: '0.45rem', fontWeight: 'bold' }}></i>}
|
|
||||||
|
|
||||||
{hubs.includes(solar_system_id.toString()) && (
|
|
||||||
<i className={PrimeIcons.MAP_MARKER} style={{ fontSize: '0.45rem', fontWeight: 'bold' }}></i>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{charactersInSystem.length > 0 && (
|
|
||||||
<div className={clsx(classes.localCounter, { ['text-amber-300']: hasUserCharacters })}>
|
|
||||||
<i className="pi pi-users" style={{ fontSize: '0.50rem' }}></i>
|
|
||||||
<span className="font-sans">{charactersInSystem.length}</span>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{visible && isShowUnsplashedSignatures && (
|
|
||||||
<div className={classes.Unsplashed}>
|
|
||||||
{unsplashedLeft.map(x => (
|
|
||||||
<UnsplashedSignature key={x.sig_id} signature={x} />
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{visible && isShowUnsplashedSignatures && (
|
|
||||||
<div className={clsx([classes.Unsplashed, classes['Unsplashed--right']])}>
|
|
||||||
{unsplashedRight.map(x => (
|
|
||||||
<UnsplashedSignature key={x.sig_id} signature={x} />
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
<div onMouseDownCapture={dbClick} className={classes.Handlers}>
|
|
||||||
<Handle
|
|
||||||
type="source"
|
|
||||||
className={clsx(classes.Handle, classes.HandleTop, {
|
|
||||||
[classes.selected]: selected,
|
|
||||||
[classes.Tick]: isThickConnections,
|
|
||||||
})}
|
|
||||||
style={{ visibility: showHandlers ? 'visible' : 'hidden' }}
|
|
||||||
position={Position.Top}
|
|
||||||
id="a"
|
|
||||||
/>
|
|
||||||
<Handle
|
|
||||||
type="source"
|
|
||||||
className={clsx(classes.Handle, classes.HandleRight, {
|
|
||||||
[classes.selected]: selected,
|
|
||||||
[classes.Tick]: isThickConnections,
|
|
||||||
})}
|
|
||||||
style={{ visibility: showHandlers ? 'visible' : 'hidden' }}
|
|
||||||
position={Position.Right}
|
|
||||||
id="b"
|
|
||||||
/>
|
|
||||||
<Handle
|
|
||||||
type="source"
|
|
||||||
className={clsx(classes.Handle, classes.HandleBottom, {
|
|
||||||
[classes.selected]: selected,
|
|
||||||
[classes.Tick]: isThickConnections,
|
|
||||||
})}
|
|
||||||
style={{ visibility: showHandlers ? 'visible' : 'hidden' }}
|
|
||||||
position={Position.Bottom}
|
|
||||||
id="c"
|
|
||||||
/>
|
|
||||||
<Handle
|
|
||||||
type="source"
|
|
||||||
className={clsx(classes.Handle, classes.HandleLeft, {
|
|
||||||
[classes.selected]: selected,
|
|
||||||
[classes.Tick]: isThickConnections,
|
|
||||||
})}
|
|
||||||
style={{ visibility: showHandlers ? 'visible' : 'hidden' }}
|
|
||||||
position={Position.Left}
|
|
||||||
id="d"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
@@ -6,7 +6,14 @@ $pastel-green: #88b04b;
|
|||||||
$pastel-yellow: #ffdd59;
|
$pastel-yellow: #ffdd59;
|
||||||
$dark-bg: #2d2d2d;
|
$dark-bg: #2d2d2d;
|
||||||
$text-color: #ffffff;
|
$text-color: #ffffff;
|
||||||
$tooltip-bg: #202020; // Dark background for tooltips
|
$tooltip-bg: #202020;
|
||||||
|
|
||||||
|
$node-bg-color: #202020;
|
||||||
|
$node-soft-bg-color: #202020;
|
||||||
|
$text-color: #ffffff;
|
||||||
|
$tag-color: #38BDF8;
|
||||||
|
$region-name: #D6D3D1;
|
||||||
|
$custom-name: #93C5FD;
|
||||||
|
|
||||||
.RootCustomNode {
|
.RootCustomNode {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -85,7 +92,7 @@ $tooltip-bg: #202020; // Dark background for tooltips
|
|||||||
box-shadow: 0 0 10px #9a1af1c2;
|
box-shadow: 0 0 10px #9a1af1c2;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip {
|
&.tooltip {
|
||||||
background-color: $tooltip-bg;
|
background-color: $tooltip-bg;
|
||||||
color: $text-color;
|
color: $text-color;
|
||||||
padding: 5px 10px;
|
padding: 5px 10px;
|
||||||
@@ -94,42 +101,59 @@ $tooltip-bg: #202020; // Dark background for tooltips
|
|||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-home {
|
&.eve-system-status-home {
|
||||||
border: 1px solid darken($eve-solar-system-status-color-home, 30%);
|
border: 1px solid var(--eve-solar-system-status-color-home-dark30);
|
||||||
background-image: linear-gradient(275deg, $eve-solar-system-status-friendly, transparent);
|
background-image: linear-gradient(
|
||||||
|
275deg,
|
||||||
|
var(--eve-solar-system-status-home),
|
||||||
|
transparent
|
||||||
|
);
|
||||||
&.selected {
|
&.selected {
|
||||||
border-color: $eve-solar-system-status-color-home;
|
border-color: var(--eve-solar-system-status-color-home);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-friendly {
|
&.eve-system-status-friendly {
|
||||||
border: 1px solid darken($eve-solar-system-status-color-friendly, 20%);
|
border: 1px solid var(--eve-solar-system-status-color-friendly-dark20);
|
||||||
background-image: linear-gradient(275deg, darken($eve-solar-system-status-friendly, 30%), transparent);
|
background-image: linear-gradient(
|
||||||
|
275deg,
|
||||||
|
var(--eve-solar-system-status-friendly-dark30),
|
||||||
|
transparent
|
||||||
|
);
|
||||||
&.selected {
|
&.selected {
|
||||||
border-color: darken($eve-solar-system-status-color-friendly, 5%);
|
border-color: var(--eve-solar-system-status-color-friendly-dark5);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-lookingFor {
|
&.eve-system-status-lookingFor {
|
||||||
border: 1px solid darken($eve-solar-system-status-color-lookingFor, 15%);
|
border: 1px solid var(--eve-solar-system-status-color-lookingFor-dark15);
|
||||||
background-image: linear-gradient(275deg, #45ff8f2f, #457fff2f);
|
background-image: linear-gradient(275deg, #45ff8f2f, #457fff2f);
|
||||||
|
|
||||||
&.selected {
|
&.selected {
|
||||||
border-color: $pastel-pink;
|
border-color: $pastel-pink;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-warning {
|
&.eve-system-status-warning {
|
||||||
background-image: linear-gradient(275deg, $eve-solar-system-status-warning, transparent);
|
background-image: linear-gradient(
|
||||||
|
275deg,
|
||||||
|
var(--eve-solar-system-status-warning),
|
||||||
|
transparent
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-dangerous {
|
&.eve-system-status-dangerous {
|
||||||
background-image: linear-gradient(275deg, $eve-solar-system-status-dangerous, transparent);
|
background-image: linear-gradient(
|
||||||
|
275deg,
|
||||||
|
var(--eve-solar-system-status-dangerous),
|
||||||
|
transparent
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
&.eve-system-status-target {
|
&.eve-system-status-target {
|
||||||
background-image: linear-gradient(275deg, $eve-solar-system-status-target, transparent);
|
background-image: linear-gradient(
|
||||||
|
275deg,
|
||||||
|
var(--eve-solar-system-status-target),
|
||||||
|
transparent
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -242,10 +266,10 @@ $tooltip-bg: #202020; // Dark background for tooltips
|
|||||||
|
|
||||||
.TagTitle {
|
.TagTitle {
|
||||||
font-size: 11px;
|
font-size: 11px;
|
||||||
font-weight: bold;
|
font-weight: medium;
|
||||||
text-shadow: 0 0 2px rgba(231, 146, 52, 0.73);
|
text-shadow: 0 0 2px rgba(231, 146, 52, 0.73);
|
||||||
|
|
||||||
color: #ffb01d;
|
color: var(--rf-tag-color, #38BDF8);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Firefox kostyl */
|
/* Firefox kostyl */
|
||||||
@@ -0,0 +1,209 @@
|
|||||||
|
import { memo } from 'react';
|
||||||
|
import { MapSolarSystemType } from '../../map.types';
|
||||||
|
import { Handle, Position, NodeProps } from 'reactflow';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import classes from './SolarSystemNodeDefault.module.scss';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { useSolarSystemNode } from '../../hooks/useSolarSystemNode';
|
||||||
|
import {
|
||||||
|
MARKER_BOOKMARK_BG_STYLES,
|
||||||
|
STATUS_CLASSES,
|
||||||
|
EFFECT_BACKGROUND_STYLES,
|
||||||
|
} from '@/hooks/Mapper/components/map/constants';
|
||||||
|
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
||||||
|
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
||||||
|
|
||||||
|
export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>) => {
|
||||||
|
const nodeVars = useSolarSystemNode(props);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{nodeVars.visible && (
|
||||||
|
<div className={classes.Bookmarks}>
|
||||||
|
{nodeVars.labelCustom !== '' && (
|
||||||
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
||||||
|
<span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] ">{nodeVars.labelCustom}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.isShattered && (
|
||||||
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.shattered)}>
|
||||||
|
<span className={clsx('pi pi-chart-pie', classes.icon)} />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.killsCount && (
|
||||||
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[nodeVars.killsActivityType!])}>
|
||||||
|
<div className={clsx(classes.BookmarkWithIcon)}>
|
||||||
|
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
||||||
|
<span className={clsx(classes.text)}>{nodeVars.killsCount}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.labelsInfo.map(x => (
|
||||||
|
<div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
|
||||||
|
{x.shortName}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
classes.RootCustomNode,
|
||||||
|
nodeVars.regionClass && classes[nodeVars.regionClass],
|
||||||
|
classes[STATUS_CLASSES[nodeVars.status]],
|
||||||
|
{
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
},
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{nodeVars.visible && (
|
||||||
|
<>
|
||||||
|
<div className={classes.HeadRow}>
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
classes.classTitle,
|
||||||
|
nodeVars.classTitleColor,
|
||||||
|
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{nodeVars.classTitle ?? '-'}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{nodeVars.tag != null && nodeVars.tag !== '' && (
|
||||||
|
<div className={clsx(classes.TagTitle, 'text-sky-400 font-medium')}>{nodeVars.tag}</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
classes.classSystemName,
|
||||||
|
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] flex-grow overflow-hidden text-ellipsis whitespace-nowrap font-sans',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{nodeVars.systemName}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{nodeVars.isWormhole && (
|
||||||
|
<div className={classes.statics}>
|
||||||
|
{nodeVars.sortedStatics.map(whClass => (
|
||||||
|
<WormholeClassComp key={whClass} id={whClass} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.effectName !== null && nodeVars.isWormhole && (
|
||||||
|
<div className={clsx(classes.effect, EFFECT_BACKGROUND_STYLES[nodeVars.effectName])} />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
|
||||||
|
{nodeVars.customName && (
|
||||||
|
<div className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] text-blue-300 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
|
||||||
|
{nodeVars.customName}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!nodeVars.isWormhole && !nodeVars.customName && (
|
||||||
|
<div className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] text-stone-300 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
|
||||||
|
{nodeVars.regionName}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.isWormhole && !nodeVars.customName && <div />}
|
||||||
|
|
||||||
|
<div className="flex items-center justify-end">
|
||||||
|
<div className="flex gap-1 items-center">
|
||||||
|
{nodeVars.locked && (
|
||||||
|
<i className={PrimeIcons.LOCK} style={{ fontSize: '0.45rem', fontWeight: 'bold' }} />
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.hubs.includes(nodeVars.solarSystemId.toString()) && (
|
||||||
|
<i className={PrimeIcons.MAP_MARKER} style={{ fontSize: '0.45rem', fontWeight: 'bold' }} />
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.charactersInSystem.length > 0 && (
|
||||||
|
<div
|
||||||
|
className={clsx(classes.localCounter, {
|
||||||
|
['text-amber-300']: nodeVars.hasUserCharacters,
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
<i className="pi pi-users" style={{ fontSize: '0.50rem' }} />
|
||||||
|
<span className="font-sans">{nodeVars.charactersInSystem.length}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{nodeVars.visible && (
|
||||||
|
<>
|
||||||
|
{nodeVars.unsplashedLeft.length > 0 && (
|
||||||
|
<div className={classes.Unsplashed}>
|
||||||
|
{nodeVars.unsplashedLeft.map(sig => (
|
||||||
|
<UnsplashedSignature key={sig.sig_id} signature={sig} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.unsplashedRight.length > 0 && (
|
||||||
|
<div className={clsx(classes.Unsplashed, classes['Unsplashed--right'])}>
|
||||||
|
{nodeVars.unsplashedRight.map(sig => (
|
||||||
|
<UnsplashedSignature key={sig.sig_id} signature={sig} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div onMouseDownCapture={nodeVars.dbClick} className={classes.Handlers}>
|
||||||
|
<Handle
|
||||||
|
type="source"
|
||||||
|
className={clsx(classes.Handle, classes.HandleTop, {
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.Tick]: nodeVars.isThickConnections,
|
||||||
|
})}
|
||||||
|
style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
|
||||||
|
position={Position.Top}
|
||||||
|
id="a"
|
||||||
|
/>
|
||||||
|
<Handle
|
||||||
|
type="source"
|
||||||
|
className={clsx(classes.Handle, classes.HandleRight, {
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.Tick]: nodeVars.isThickConnections,
|
||||||
|
})}
|
||||||
|
style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
|
||||||
|
position={Position.Right}
|
||||||
|
id="b"
|
||||||
|
/>
|
||||||
|
<Handle
|
||||||
|
type="source"
|
||||||
|
className={clsx(classes.Handle, classes.HandleBottom, {
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.Tick]: nodeVars.isThickConnections,
|
||||||
|
})}
|
||||||
|
style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
|
||||||
|
position={Position.Bottom}
|
||||||
|
id="c"
|
||||||
|
/>
|
||||||
|
<Handle
|
||||||
|
type="source"
|
||||||
|
className={clsx(classes.Handle, classes.HandleLeft, {
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.Tick]: nodeVars.isThickConnections,
|
||||||
|
})}
|
||||||
|
style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
|
||||||
|
position={Position.Left}
|
||||||
|
id="d"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
SolarSystemNodeDefault.displayName = 'SolarSystemNodeDefault';
|
||||||
@@ -0,0 +1,91 @@
|
|||||||
|
@import './SolarSystemNodeDefault.module.scss';
|
||||||
|
|
||||||
|
/* ---------------------------
|
||||||
|
Only override what's different
|
||||||
|
--------------------------- */
|
||||||
|
|
||||||
|
/* 1) .RootCustomNode:
|
||||||
|
- new background-color using CSS var
|
||||||
|
- plus color, font-family, and font-weight */
|
||||||
|
.RootCustomNode {
|
||||||
|
background-color: var(--rf-node-bg-color, #202020) !important;
|
||||||
|
color: var(--rf-text-color, #ffffff);
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 2) .Bookmarks:
|
||||||
|
- add var-based font family/weight
|
||||||
|
*/
|
||||||
|
.Bookmarks {
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 3) .HeadRow, .classTitle, .classSystemName:
|
||||||
|
- add new references to var-based font family/weight
|
||||||
|
*/
|
||||||
|
.HeadRow {
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
|
||||||
|
.classTitle {
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
@-moz-document url-prefix() {
|
||||||
|
.classSystemName {
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.classSystemName {
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 4) .BottomRow:
|
||||||
|
- introduces .tagTitle, .regionName, .customName, .localCounter
|
||||||
|
referencing new CSS variables */
|
||||||
|
.BottomRow {
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
|
||||||
|
.tagTitle {
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: medium;
|
||||||
|
text-shadow: 0 0 2px rgba(231, 146, 52, 0.73);
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
color: var(--rf-tag-color, #38BDF8);
|
||||||
|
}
|
||||||
|
|
||||||
|
.regionName {
|
||||||
|
color: var(--rf-region-name, #D6D3D1);
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.customName {
|
||||||
|
color: var(--rf-custom-name, #93C5FD);
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.localCounter {
|
||||||
|
display: flex;
|
||||||
|
color: var(--rf-has-user-characters, #fbbf24);
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
gap: 2px;
|
||||||
|
|
||||||
|
.hasUserCharacters {
|
||||||
|
color: var(--rf-has-user-characters, #fbbf24);
|
||||||
|
font-family: var(--rf-node-font-family, inherit) !important;
|
||||||
|
font-weight: var(--rf-node-font-weight, inherit) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,219 @@
|
|||||||
|
import { memo } from 'react';
|
||||||
|
import { MapSolarSystemType } from '../../map.types';
|
||||||
|
import { Handle, Position, NodeProps } from 'reactflow';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import classes from './SolarSystemNodeTheme.module.scss';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { useSolarSystemNode } from '../../hooks/useSolarSystemNode';
|
||||||
|
import {
|
||||||
|
MARKER_BOOKMARK_BG_STYLES,
|
||||||
|
STATUS_CLASSES,
|
||||||
|
EFFECT_BACKGROUND_STYLES,
|
||||||
|
} from '@/hooks/Mapper/components/map/constants';
|
||||||
|
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
||||||
|
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
||||||
|
|
||||||
|
export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>) => {
|
||||||
|
const nodeVars = useSolarSystemNode(props);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{nodeVars.visible && (
|
||||||
|
<div className={classes.Bookmarks}>
|
||||||
|
{nodeVars.labelCustom !== '' && (
|
||||||
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.custom)}>
|
||||||
|
<span className="[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] ">{nodeVars.labelCustom}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.isShattered && (
|
||||||
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES.shattered)}>
|
||||||
|
<span className={clsx('pi pi-chart-pie', classes.icon)} />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.killsCount && (
|
||||||
|
<div className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[nodeVars.killsActivityType!])}>
|
||||||
|
<div className={clsx(classes.BookmarkWithIcon)}>
|
||||||
|
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
||||||
|
<span className={clsx(classes.text)}>{nodeVars.killsCount}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.labelsInfo.map(x => (
|
||||||
|
<div key={x.id} className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[x.id])}>
|
||||||
|
{x.shortName}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
classes.RootCustomNode,
|
||||||
|
nodeVars.regionClass && classes[nodeVars.regionClass],
|
||||||
|
classes[STATUS_CLASSES[nodeVars.status]],
|
||||||
|
{
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
},
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{nodeVars.visible && (
|
||||||
|
<>
|
||||||
|
<div className={classes.HeadRow}>
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
classes.classTitle,
|
||||||
|
nodeVars.classTitleColor,
|
||||||
|
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)]',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{nodeVars.classTitle ?? '-'}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{nodeVars.tag != null && nodeVars.tag !== '' && (
|
||||||
|
<div className={clsx(classes.TagTitle)}>{nodeVars.tag}</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
classes.classSystemName,
|
||||||
|
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] flex-grow overflow-hidden text-ellipsis whitespace-nowrap',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{nodeVars.systemName}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{nodeVars.isWormhole && (
|
||||||
|
<div className={classes.statics}>
|
||||||
|
{nodeVars.sortedStatics.map(whClass => (
|
||||||
|
<WormholeClassComp key={whClass} id={whClass} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.effectName !== null && nodeVars.isWormhole && (
|
||||||
|
<div className={clsx(classes.effect, EFFECT_BACKGROUND_STYLES[nodeVars.effectName])} />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className={clsx(classes.BottomRow, 'flex items-center justify-between')}>
|
||||||
|
{nodeVars.customName && (
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
classes.CustomName,
|
||||||
|
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] whitespace-nowrap overflow-hidden text-ellipsis mr-0.5',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{nodeVars.customName}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!nodeVars.isWormhole && !nodeVars.customName && (
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
classes.RegionName,
|
||||||
|
'[text-shadow:_0_1px_0_rgb(0_0_0_/_40%)] whitespace-nowrap overflow-hidden text-ellipsis mr-0.5',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{nodeVars.regionName}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.isWormhole && !nodeVars.customName && <div />}
|
||||||
|
|
||||||
|
<div className="flex items-center justify-end">
|
||||||
|
<div className="flex gap-1 items-center">
|
||||||
|
{nodeVars.locked && (
|
||||||
|
<i className={PrimeIcons.LOCK} style={{ fontSize: '0.45rem', fontWeight: 'bold' }} />
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.hubs.includes(nodeVars.solarSystemId.toString()) && (
|
||||||
|
<i className={PrimeIcons.MAP_MARKER} style={{ fontSize: '0.45rem', fontWeight: 'bold' }} />
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.charactersInSystem.length > 0 && (
|
||||||
|
<div
|
||||||
|
className={clsx(classes.localCounter, {
|
||||||
|
[classes.hasUserCharacters]: nodeVars.hasUserCharacters,
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
<i className="pi pi-users" style={{ fontSize: '0.50rem' }} />
|
||||||
|
<span className="font-sans">{nodeVars.charactersInSystem.length}</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{nodeVars.visible && (
|
||||||
|
<>
|
||||||
|
{nodeVars.unsplashedLeft.length > 0 && (
|
||||||
|
<div className={classes.Unsplashed}>
|
||||||
|
{nodeVars.unsplashedLeft.map(sig => (
|
||||||
|
<UnsplashedSignature key={sig.sig_id} signature={sig} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{nodeVars.unsplashedRight.length > 0 && (
|
||||||
|
<div className={clsx(classes.Unsplashed, classes['Unsplashed--right'])}>
|
||||||
|
{nodeVars.unsplashedRight.map(sig => (
|
||||||
|
<UnsplashedSignature key={sig.sig_id} signature={sig} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div onMouseDownCapture={nodeVars.dbClick} className={classes.Handlers}>
|
||||||
|
<Handle
|
||||||
|
type="source"
|
||||||
|
className={clsx(classes.Handle, classes.HandleTop, {
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.Tick]: nodeVars.isThickConnections,
|
||||||
|
})}
|
||||||
|
style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
|
||||||
|
position={Position.Top}
|
||||||
|
id="a"
|
||||||
|
/>
|
||||||
|
<Handle
|
||||||
|
type="source"
|
||||||
|
className={clsx(classes.Handle, classes.HandleRight, {
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.Tick]: nodeVars.isThickConnections,
|
||||||
|
})}
|
||||||
|
style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
|
||||||
|
position={Position.Right}
|
||||||
|
id="b"
|
||||||
|
/>
|
||||||
|
<Handle
|
||||||
|
type="source"
|
||||||
|
className={clsx(classes.Handle, classes.HandleBottom, {
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.Tick]: nodeVars.isThickConnections,
|
||||||
|
})}
|
||||||
|
style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
|
||||||
|
position={Position.Bottom}
|
||||||
|
id="c"
|
||||||
|
/>
|
||||||
|
<Handle
|
||||||
|
type="source"
|
||||||
|
className={clsx(classes.Handle, classes.HandleLeft, {
|
||||||
|
[classes.selected]: nodeVars.selected,
|
||||||
|
[classes.Tick]: nodeVars.isThickConnections,
|
||||||
|
})}
|
||||||
|
style={{ visibility: nodeVars.showHandlers ? 'visible' : 'hidden' }}
|
||||||
|
position={Position.Left}
|
||||||
|
id="d"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
SolarSystemNodeTheme.displayName = 'SolarSystemNodeTheme';
|
||||||
@@ -1 +1,2 @@
|
|||||||
export * from './SolarSystemNode';
|
export * from './SolarSystemNodeDefault';
|
||||||
|
export * from './SolarSystemNodeTheme';
|
||||||
|
|||||||
@@ -9,10 +9,14 @@
|
|||||||
width: 13px;
|
width: 13px;
|
||||||
height: 4px;
|
height: 4px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
color: #ffffff;
|
color: var(--text-color);
|
||||||
font-size: 8px;
|
font-size: 8px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
font-weight: bolder;
|
font-weight: bolder;
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
& > .Eol {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,8 +8,8 @@ import { WORMHOLE_CLASS_STYLES, WORMHOLES_ADDITIONAL_INFO } from '@/hooks/Mapper
|
|||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import { renderInfoColumn } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/renders';
|
import { renderInfoColumn } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/renders';
|
||||||
|
import { K162_TYPES_MAP } from '@/hooks/Mapper/constants.ts';
|
||||||
import { k162Types } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureK162TypeSelect';
|
import { parseSignatureCustomInfo } from '@/hooks/Mapper/helpers/parseSignatureCustomInfo.ts';
|
||||||
|
|
||||||
interface UnsplashedSignatureProps {
|
interface UnsplashedSignatureProps {
|
||||||
signature: SystemSignature;
|
signature: SystemSignature;
|
||||||
@@ -22,17 +22,22 @@ export const UnsplashedSignature = ({ signature }: UnsplashedSignatureProps) =>
|
|||||||
const whData = useMemo(() => wormholesData[signature.type], [signature.type, wormholesData]);
|
const whData = useMemo(() => wormholesData[signature.type], [signature.type, wormholesData]);
|
||||||
const whClass = useMemo(() => (whData ? WORMHOLES_ADDITIONAL_INFO[whData.dest] : null), [whData]);
|
const whClass = useMemo(() => (whData ? WORMHOLES_ADDITIONAL_INFO[whData.dest] : null), [whData]);
|
||||||
|
|
||||||
const k162TypeOption = useMemo(() => {
|
const customInfo = useMemo(() => {
|
||||||
if (!signature.custom_info) {
|
return parseSignatureCustomInfo(signature.custom_info);
|
||||||
return null;
|
|
||||||
}
|
|
||||||
const customInfo = JSON.parse(signature.custom_info);
|
|
||||||
if (!customInfo.k162Type) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
return k162Types.find(x => x.value === customInfo.k162Type);
|
|
||||||
}, [signature]);
|
}, [signature]);
|
||||||
|
|
||||||
|
const k162TypeOption = useMemo(() => {
|
||||||
|
if (!customInfo?.k162Type) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return K162_TYPES_MAP[customInfo.k162Type];
|
||||||
|
}, [customInfo]);
|
||||||
|
|
||||||
|
const isEOL = useMemo(() => {
|
||||||
|
return customInfo?.isEOL;
|
||||||
|
}, [customInfo]);
|
||||||
|
|
||||||
const whClassStyle = useMemo(() => {
|
const whClassStyle = useMemo(() => {
|
||||||
if (signature.type === 'K162' && k162TypeOption) {
|
if (signature.type === 'K162' && k162TypeOption) {
|
||||||
const k162Data = wormholesData[k162TypeOption.whClassName];
|
const k162Data = wormholesData[k162TypeOption.whClassName];
|
||||||
@@ -45,19 +50,19 @@ export const UnsplashedSignature = ({ signature }: UnsplashedSignatureProps) =>
|
|||||||
return (
|
return (
|
||||||
<WdTooltipWrapper
|
<WdTooltipWrapper
|
||||||
className={clsx(classes.Signature)}
|
className={clsx(classes.Signature)}
|
||||||
|
// @ts-ignore
|
||||||
content={
|
content={
|
||||||
(
|
<div className="flex flex-col gap-1">
|
||||||
<div className="flex flex-col gap-1">
|
<InfoDrawer title={<b className="text-slate-50">{signature.eve_id}</b>}>
|
||||||
<InfoDrawer title={<b className="text-slate-50">{signature.eve_id}</b>}>
|
{renderInfoColumn(signature)}
|
||||||
{renderInfoColumn(signature)}
|
</InfoDrawer>
|
||||||
</InfoDrawer>
|
</div>
|
||||||
</div>
|
|
||||||
) as React.ReactNode
|
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
<div className={clsx(classes.Box, whClassStyle)}>
|
<div className={clsx(classes.Box, whClassStyle)}>
|
||||||
<svg width="13" height="4" viewBox="0 0 13 4" xmlns="http://www.w3.org/2000/svg">
|
<svg width="13" height="8" viewBox="0 0 13 8" xmlns="http://www.w3.org/2000/svg">
|
||||||
<rect width="13" height="4" rx="2" className={whClassStyle} fill="currentColor" />
|
<rect y="1" width="13" height="4" rx="2" className={whClassStyle} fill="currentColor" />
|
||||||
|
{isEOL && <rect x="4" width="5" height="6" rx="1" className={clsx(classes.Eol)} fill="#a153ac" />}
|
||||||
</svg>
|
</svg>
|
||||||
</div>
|
</div>
|
||||||
</WdTooltipWrapper>
|
</WdTooltipWrapper>
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
import { SolarSystemNodeDefault, SolarSystemNodeTheme } from '../components/SolarSystemNode';
|
||||||
|
import type { NodeProps } from 'reactflow';
|
||||||
|
import type { ComponentType } from 'react';
|
||||||
|
import { MapSolarSystemType } from '../map.types';
|
||||||
|
import { ConnectionMode } from 'reactflow';
|
||||||
|
|
||||||
|
export type SolarSystemNodeComponent = ComponentType<NodeProps<MapSolarSystemType>>;
|
||||||
|
|
||||||
|
interface ThemeBehavior {
|
||||||
|
isPanAndDrag: boolean;
|
||||||
|
nodeComponent: SolarSystemNodeComponent;
|
||||||
|
connectionMode: ConnectionMode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const THEME_BEHAVIORS: {
|
||||||
|
[key: string]: ThemeBehavior;
|
||||||
|
} = {
|
||||||
|
default: {
|
||||||
|
isPanAndDrag: false,
|
||||||
|
nodeComponent: SolarSystemNodeDefault,
|
||||||
|
connectionMode: ConnectionMode.Loose,
|
||||||
|
},
|
||||||
|
pathfinder: {
|
||||||
|
isPanAndDrag: true,
|
||||||
|
nodeComponent: SolarSystemNodeTheme,
|
||||||
|
connectionMode: ConnectionMode.Loose,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export function getBehaviorForTheme(themeName: string) {
|
||||||
|
return THEME_BEHAVIORS[themeName] ?? THEME_BEHAVIORS.default;
|
||||||
|
}
|
||||||
@@ -42,7 +42,7 @@ export const useMapUpdateSystems = () => {
|
|||||||
return newSystem;
|
return newSystem;
|
||||||
});
|
});
|
||||||
|
|
||||||
update({ systems: out });
|
update({ systems: out }, true);
|
||||||
},
|
},
|
||||||
[rf, update],
|
[rf, update],
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -0,0 +1,44 @@
|
|||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { BackgroundVariant } from 'reactflow';
|
||||||
|
|
||||||
|
export function useBackgroundVars(themeName?: string) {
|
||||||
|
const [variant, setVariant] = useState<BackgroundVariant>(BackgroundVariant.Dots);
|
||||||
|
const [gap, setGap] = useState<number>(16);
|
||||||
|
const [size, setSize] = useState<number>(1);
|
||||||
|
const [color, setColor] = useState('#81818b');
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
// match any element whose entire `class` attribute ends with "-theme"
|
||||||
|
let themeEl = document.querySelector('[class$="-theme"]');
|
||||||
|
|
||||||
|
// If none is found, fall back to the <html> element
|
||||||
|
if (!themeEl) {
|
||||||
|
themeEl = document.documentElement;
|
||||||
|
}
|
||||||
|
|
||||||
|
const style = getComputedStyle(themeEl as HTMLElement);
|
||||||
|
|
||||||
|
const rawVariant = style.getPropertyValue('--rf-bg-variant').replace(/['"]/g, '').trim().toLowerCase();
|
||||||
|
let finalVariant: BackgroundVariant = BackgroundVariant.Dots;
|
||||||
|
|
||||||
|
if (rawVariant === 'lines') {
|
||||||
|
finalVariant = BackgroundVariant.Lines;
|
||||||
|
} else if (rawVariant === 'cross') {
|
||||||
|
finalVariant = BackgroundVariant.Cross;
|
||||||
|
}
|
||||||
|
|
||||||
|
const cssVarGap = style.getPropertyValue('--rf-bg-gap');
|
||||||
|
const cssVarSize = style.getPropertyValue('--rf-bg-size');
|
||||||
|
const cssColor = style.getPropertyValue('--rf-bg-pattern-color');
|
||||||
|
|
||||||
|
const gapNum = parseInt(cssVarGap, 10) || 16;
|
||||||
|
const sizeNum = parseInt(cssVarSize, 10) || 1;
|
||||||
|
|
||||||
|
setVariant(finalVariant);
|
||||||
|
setGap(gapNum);
|
||||||
|
setSize(sizeNum);
|
||||||
|
setColor(cssColor);
|
||||||
|
}, [themeName]);
|
||||||
|
|
||||||
|
return { variant, gap, size, color };
|
||||||
|
}
|
||||||
@@ -70,7 +70,7 @@ export const useMapHandlers = (ref: ForwardedRef<MapHandlers>, onSelectionChange
|
|||||||
setTimeout(() => addConnections(data as CommandAddConnections), 100);
|
setTimeout(() => addConnections(data as CommandAddConnections), 100);
|
||||||
break;
|
break;
|
||||||
case Commands.removeConnections:
|
case Commands.removeConnections:
|
||||||
removeConnections(data as CommandRemoveConnections);
|
setTimeout(() => removeConnections(data as CommandRemoveConnections), 100);
|
||||||
break;
|
break;
|
||||||
case Commands.charactersUpdated:
|
case Commands.charactersUpdated:
|
||||||
charactersUpdated(data as CommandCharactersUpdated);
|
charactersUpdated(data as CommandCharactersUpdated);
|
||||||
|
|||||||
@@ -0,0 +1,254 @@
|
|||||||
|
import { useMemo } from 'react';
|
||||||
|
import { MapSolarSystemType } from '../map.types';
|
||||||
|
import { NodeProps } from 'reactflow';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
|
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider';
|
||||||
|
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick';
|
||||||
|
import { REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
|
||||||
|
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace';
|
||||||
|
import { getSystemClassStyles, prepareUnsplashedChunks } from '@/hooks/Mapper/components/map/helpers';
|
||||||
|
import { sortWHClasses } from '@/hooks/Mapper/helpers';
|
||||||
|
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager';
|
||||||
|
import { CharacterTypeRaw, OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
import { LABELS_INFO, LABELS_ORDER } from '@/hooks/Mapper/components/map/constants';
|
||||||
|
|
||||||
|
function getActivityType(count: number) {
|
||||||
|
if (count <= 5) return 'activityNormal';
|
||||||
|
if (count <= 30) return 'activityWarn';
|
||||||
|
return 'activityDanger';
|
||||||
|
}
|
||||||
|
|
||||||
|
const SpaceToClass: Record<string, string> = {
|
||||||
|
[Spaces.Caldari]: 'Caldaria',
|
||||||
|
[Spaces.Matar]: 'Mataria',
|
||||||
|
[Spaces.Amarr]: 'Amarria',
|
||||||
|
[Spaces.Gallente]: 'Gallente',
|
||||||
|
};
|
||||||
|
|
||||||
|
function sortedLabels(labels: string[]) {
|
||||||
|
if (!labels) return [];
|
||||||
|
return LABELS_ORDER.filter(x => labels.includes(x)).map(x => LABELS_INFO[x]);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useSolarSystemNode(props: NodeProps<MapSolarSystemType>) {
|
||||||
|
const { id, data, selected } = props;
|
||||||
|
const {
|
||||||
|
system_static_info,
|
||||||
|
system_signatures,
|
||||||
|
locked,
|
||||||
|
name,
|
||||||
|
tag,
|
||||||
|
status,
|
||||||
|
labels,
|
||||||
|
temporary_name,
|
||||||
|
linked_sig_eve_id: linkedSigEveId = '',
|
||||||
|
} = data;
|
||||||
|
|
||||||
|
const {
|
||||||
|
system_class,
|
||||||
|
security,
|
||||||
|
class_title,
|
||||||
|
solar_system_id,
|
||||||
|
statics,
|
||||||
|
effect_name,
|
||||||
|
region_name,
|
||||||
|
region_id,
|
||||||
|
is_shattered,
|
||||||
|
solar_system_name,
|
||||||
|
} = system_static_info;
|
||||||
|
|
||||||
|
const {
|
||||||
|
interfaceSettings,
|
||||||
|
data: { systemSignatures: mapSystemSignatures },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const { isShowUnsplashedSignatures } = interfaceSettings;
|
||||||
|
const isTempSystemNameEnabled = useMapGetOption('show_temp_system_name') === 'true';
|
||||||
|
const isShowLinkedSigId = useMapGetOption('show_linked_signature_id') === 'true';
|
||||||
|
const isShowLinkedSigIdTempName = useMapGetOption('show_linked_signature_id_temp_name') === 'true';
|
||||||
|
|
||||||
|
const {
|
||||||
|
data: {
|
||||||
|
characters,
|
||||||
|
presentCharacters,
|
||||||
|
wormholesData,
|
||||||
|
hubs,
|
||||||
|
kills,
|
||||||
|
userCharacters,
|
||||||
|
isConnecting,
|
||||||
|
hoverNodeId,
|
||||||
|
visibleNodes,
|
||||||
|
showKSpaceBG,
|
||||||
|
isThickConnections,
|
||||||
|
},
|
||||||
|
outCommand,
|
||||||
|
} = useMapState();
|
||||||
|
|
||||||
|
const visible = useMemo(() => visibleNodes.has(id), [id, visibleNodes]);
|
||||||
|
|
||||||
|
const systemSignatures = useMemo(
|
||||||
|
() => mapSystemSignatures[solar_system_id] || system_signatures,
|
||||||
|
[system_signatures, solar_system_id, mapSystemSignatures],
|
||||||
|
);
|
||||||
|
|
||||||
|
const charactersInSystem = useMemo(() => {
|
||||||
|
return characters.filter(c => c.location?.solar_system_id === solar_system_id).filter(c => c.online);
|
||||||
|
// eslint-disable-next-line
|
||||||
|
}, [characters, presentCharacters, solar_system_id]);
|
||||||
|
|
||||||
|
const isWormhole = isWormholeSpace(system_class);
|
||||||
|
|
||||||
|
const classTitleColor = useMemo(
|
||||||
|
() => getSystemClassStyles({ systemClass: system_class, security }),
|
||||||
|
[security, system_class],
|
||||||
|
);
|
||||||
|
|
||||||
|
const sortedStatics = useMemo(() => sortWHClasses(wormholesData, statics), [wormholesData, statics]);
|
||||||
|
|
||||||
|
const linkedSigPrefix = useMemo(() => (linkedSigEveId ? linkedSigEveId.split('-')[0] : null), [linkedSigEveId]);
|
||||||
|
|
||||||
|
const labelsManager = useMemo(() => new LabelsManager(labels ?? ''), [labels]);
|
||||||
|
const labelsInfo = useMemo(() => sortedLabels(labelsManager.list), [labelsManager]);
|
||||||
|
const labelCustom = useMemo(() => {
|
||||||
|
if (isShowLinkedSigId && linkedSigPrefix) {
|
||||||
|
return labelsManager.customLabel ? `${linkedSigPrefix}・${labelsManager.customLabel}` : linkedSigPrefix;
|
||||||
|
}
|
||||||
|
return labelsManager.customLabel;
|
||||||
|
}, [linkedSigPrefix, isShowLinkedSigId, labelsManager]);
|
||||||
|
|
||||||
|
const killsCount = useMemo(() => kills[solar_system_id] ?? null, [kills, solar_system_id]);
|
||||||
|
const killsActivityType = killsCount ? getActivityType(killsCount) : null;
|
||||||
|
|
||||||
|
const hasUserCharacters = useMemo(() => {
|
||||||
|
return charactersInSystem.some(x => userCharacters.includes(x.eve_id));
|
||||||
|
}, [charactersInSystem, userCharacters]);
|
||||||
|
|
||||||
|
const dbClick = useDoubleClick(() => {
|
||||||
|
outCommand({
|
||||||
|
type: OutCommand.openSettings,
|
||||||
|
data: { system_id: solar_system_id.toString() },
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const showHandlers = isConnecting || hoverNodeId === id;
|
||||||
|
|
||||||
|
const space = showKSpaceBG ? REGIONS_MAP[region_id] : '';
|
||||||
|
const regionClass = showKSpaceBG ? SpaceToClass[space] : null;
|
||||||
|
|
||||||
|
const temporaryName = useMemo(() => {
|
||||||
|
if (!isTempSystemNameEnabled) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isShowLinkedSigIdTempName && linkedSigPrefix) {
|
||||||
|
return temporary_name ? `${linkedSigPrefix}・${temporary_name}` : `${linkedSigPrefix}・${solar_system_name}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return temporary_name;
|
||||||
|
}, [isShowLinkedSigIdTempName, isTempSystemNameEnabled, linkedSigPrefix, solar_system_name, temporary_name]);
|
||||||
|
|
||||||
|
const systemName = useMemo(() => {
|
||||||
|
if (isTempSystemNameEnabled && temporaryName) {
|
||||||
|
return temporaryName;
|
||||||
|
}
|
||||||
|
return solar_system_name;
|
||||||
|
}, [isTempSystemNameEnabled, solar_system_name, temporaryName]);
|
||||||
|
|
||||||
|
const customName = (isTempSystemNameEnabled && temporaryName && name) || (solar_system_name !== name && name) || null;
|
||||||
|
|
||||||
|
const [unsplashedLeft, unsplashedRight] = useMemo(() => {
|
||||||
|
if (!isShowUnsplashedSignatures) {
|
||||||
|
return [[], []];
|
||||||
|
}
|
||||||
|
return prepareUnsplashedChunks(
|
||||||
|
systemSignatures
|
||||||
|
.filter(s => s.group === 'Wormhole' && !s.linked_system)
|
||||||
|
.map(s => ({
|
||||||
|
eve_id: s.eve_id,
|
||||||
|
type: s.type,
|
||||||
|
custom_info: s.custom_info,
|
||||||
|
})),
|
||||||
|
);
|
||||||
|
}, [isShowUnsplashedSignatures, systemSignatures]);
|
||||||
|
|
||||||
|
const nodeVars = {
|
||||||
|
id,
|
||||||
|
selected,
|
||||||
|
|
||||||
|
visible,
|
||||||
|
isWormhole,
|
||||||
|
classTitleColor,
|
||||||
|
killsCount,
|
||||||
|
killsActivityType,
|
||||||
|
hasUserCharacters,
|
||||||
|
showHandlers,
|
||||||
|
regionClass,
|
||||||
|
systemName,
|
||||||
|
customName,
|
||||||
|
labelCustom,
|
||||||
|
isShattered: is_shattered,
|
||||||
|
tag,
|
||||||
|
status,
|
||||||
|
labelsInfo,
|
||||||
|
dbClick,
|
||||||
|
sortedStatics,
|
||||||
|
effectName: effect_name,
|
||||||
|
regionName: region_name,
|
||||||
|
solarSystemId: solar_system_id,
|
||||||
|
solarSystemName: solar_system_name,
|
||||||
|
locked,
|
||||||
|
hubs,
|
||||||
|
name: name,
|
||||||
|
isConnecting,
|
||||||
|
hoverNodeId,
|
||||||
|
charactersInSystem,
|
||||||
|
unsplashedLeft,
|
||||||
|
unsplashedRight,
|
||||||
|
isThickConnections,
|
||||||
|
classTitle: class_title,
|
||||||
|
temporaryName: temporary_name,
|
||||||
|
};
|
||||||
|
|
||||||
|
return nodeVars;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SolarSystemNodeVars {
|
||||||
|
id: string;
|
||||||
|
selected: boolean;
|
||||||
|
visible: boolean;
|
||||||
|
isWormhole: boolean;
|
||||||
|
classTitleColor: string | null;
|
||||||
|
killsCount: number | null;
|
||||||
|
killsActivityType: string | null;
|
||||||
|
hasUserCharacters: boolean;
|
||||||
|
showHandlers: boolean;
|
||||||
|
regionClass: string | null;
|
||||||
|
systemName: string;
|
||||||
|
customName?: string | null;
|
||||||
|
labelCustom: string | null;
|
||||||
|
isShattered: boolean;
|
||||||
|
tag?: string | null;
|
||||||
|
status?: number;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
labelsInfo: Array<any>;
|
||||||
|
dbClick: (event?: void) => void;
|
||||||
|
sortedStatics: Array<string | number>;
|
||||||
|
effectName: string | null;
|
||||||
|
regionName: string | null;
|
||||||
|
solarSystemId: number;
|
||||||
|
solarSystemName: string | null;
|
||||||
|
locked: boolean;
|
||||||
|
hubs: string[] | number[];
|
||||||
|
name: string | null;
|
||||||
|
isConnecting: boolean;
|
||||||
|
hoverNodeId: string | null;
|
||||||
|
charactersInSystem: Array<CharacterTypeRaw>;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
unsplashedLeft: Array<any>;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
unsplashedRight: Array<any>;
|
||||||
|
isThickConnections: boolean;
|
||||||
|
classTitle: string | null;
|
||||||
|
temporaryName?: string | null;
|
||||||
|
}
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
@import './eve-common-variables';
|
||||||
|
@import './eve-common';
|
||||||
|
|
||||||
|
.default-theme {
|
||||||
|
--rf-bg-color: #0C0A09;
|
||||||
|
--rf-soft-bg-color: #171717;
|
||||||
|
|
||||||
|
--rf-node-bg-color: #202020;
|
||||||
|
--rf-node-soft-bg-color: #202020;
|
||||||
|
--rf-text-color: #ffffff;
|
||||||
|
--rf-tag-color: #38BDF8;
|
||||||
|
--rf-region-name: #D6D3D1;
|
||||||
|
--rf-custom-name: #93C5FD;
|
||||||
|
|
||||||
|
|
||||||
|
--rf-bg-variant: "dots";
|
||||||
|
--rf-bg-gap: 15;
|
||||||
|
--rf-bg-size: 1;
|
||||||
|
--rf-bg-pattern-color: #81818a;
|
||||||
|
|
||||||
|
--pastel-blue: #5a7d9a;
|
||||||
|
--pastel-pink: #d291bc;
|
||||||
|
--pastel-green: #88b04b;
|
||||||
|
--pastel-yellow: #ffdd59;
|
||||||
|
|
||||||
|
--dark-bg: #2d2d2d;
|
||||||
|
--text-color: #ffffff;
|
||||||
|
--tooltip-bg: #202020;
|
||||||
|
|
||||||
|
--window-corner: #72716f;
|
||||||
|
}
|
||||||
@@ -1,78 +1,121 @@
|
|||||||
$eve-link-color-default: #333;
|
|
||||||
$eve-link-color-top-mass-0: #333;
|
|
||||||
$eve-link-color-top-mass-1: #5a4520;
|
|
||||||
$eve-link-color-top-mass-2: #672c2c;
|
|
||||||
$eve-link-color-middle-mass-0: #333;
|
|
||||||
$eve-link-color-middle-mass-1: #333;
|
|
||||||
$eve-link-color-middle-mass-2: #333;
|
|
||||||
$eve-link-color-middle-time-0: #5c5c5c;
|
|
||||||
$eve-link-color-middle-time-1: #ff00cd;
|
|
||||||
$eve-link-color-middle-time-1-border: #99f3ff;
|
|
||||||
|
|
||||||
$eve-link-color-top-mass-1-time-1: #796300;
|
$friendlyBase: #3bbd39;
|
||||||
$eve-link-color-top-mass-2-time-1: #8c1717;
|
$friendlyAlpha: #3bbd3952;
|
||||||
$eve-link-color-temp: orange;
|
$friendlyDark20: darken($friendlyBase, 20%);
|
||||||
|
$friendlyDark30: darken($friendlyBase, 30%);
|
||||||
|
$friendlyDark5: darken($friendlyBase, 5%);
|
||||||
|
|
||||||
$eve-effect-pulsar: #40aef5;
|
$lookingForBase: #43c2fd;
|
||||||
$eve-effect-magnetar: #f058f8;
|
$lookingForAlpha: rgba(67, 176, 253, 0.48);
|
||||||
$eve-effect-wolfRayet: #ef7843;
|
$lookingForDark15: darken($lookingForBase, 15%);
|
||||||
$eve-effect-blackHole: #1b1b1b;
|
|
||||||
$eve-effect-cataclysmicVariable: #ffea90;
|
|
||||||
$eve-effect-redGiant: #fd3c3c;
|
|
||||||
$eve-effect-dazhLiminalityLocus: #ff6464;
|
|
||||||
$eve-effect-imperialStellarObservatory: #6991ce;
|
|
||||||
$eve-effect-stateStellarObservatory: #6991ce;
|
|
||||||
$eve-effect-republicStellarObservatory: #6991ce;
|
|
||||||
$eve-effect-federalStellarObservatory: #6991ce;
|
|
||||||
|
|
||||||
$eve-wh-type-color-high: #5dffd2;
|
$homeBase: rgb(197, 253, 67);
|
||||||
$eve-wh-type-color-low: #f79400;
|
$homeAlpha: rgba(197, 253, 67, 0.32);
|
||||||
$eve-wh-type-color-null: #fc3c3c;
|
$homeDark30: darken($homeBase, 30%);
|
||||||
$eve-wh-type-color-c1: #69bfce;
|
|
||||||
$eve-wh-type-color-c2: #6991ce;
|
|
||||||
$eve-wh-type-color-c3: #a8cb70;
|
|
||||||
$eve-wh-type-color-c4: #e39c68;
|
|
||||||
$eve-wh-type-color-c5: #de8686;
|
|
||||||
$eve-wh-type-color-c6: #e76363;
|
|
||||||
$eve-wh-type-color-c13: #988cb5;
|
|
||||||
$eve-wh-type-color-drifter: #ff44f6;
|
|
||||||
$eve-wh-type-color-thera: #ffffff;
|
|
||||||
$eve-wh-type-color-zarzakh: #212121;
|
|
||||||
|
|
||||||
$eve-security-color-10: #2c74df;
|
|
||||||
$eve-security-color-09: #3998e8;
|
|
||||||
$eve-security-color-08: #4dcbf5;
|
|
||||||
$eve-security-color-07: #60d8a2;
|
|
||||||
$eve-security-color-06: #71e454;
|
|
||||||
$eve-security-color-05: #f2fc81;
|
|
||||||
$eve-security-color-04: #d96c07;
|
|
||||||
$eve-security-color-03: #cb440f;
|
|
||||||
$eve-security-color-02: #b91117;
|
|
||||||
$eve-security-color-01: #732020;
|
|
||||||
$eve-security-color-00: #8b3263;
|
|
||||||
$eve-security-color-m-01: #8b3263;
|
|
||||||
$eve-security-color-m-02: #8b3263;
|
|
||||||
$eve-security-color-m-03: #8b3263;
|
|
||||||
$eve-security-color-m-04: #8b3263;
|
|
||||||
$eve-security-color-m-05: #8b3263;
|
|
||||||
$eve-security-color-m-06: #8b3263;
|
|
||||||
$eve-security-color-m-07: #8b3263;
|
|
||||||
$eve-security-color-m-08: #8b3263;
|
|
||||||
$eve-security-color-m-09: #8b3263;
|
|
||||||
$eve-security-color-m-10: #8b3263;
|
|
||||||
|
|
||||||
$eve-solar-system-status-unknown: transparent;
|
:root {
|
||||||
$eve-solar-system-status-friendly: #3bbd3952;
|
--pastel-blue: #5a7d9a;
|
||||||
$eve-solar-system-status-warning: #906518a6;
|
--pastel-pink: #d291bc;
|
||||||
$eve-solar-system-status-target: #b439ff6b;
|
--pastel-green: #88b04b;
|
||||||
$eve-solar-system-status-dangerous: #d54040;
|
--pastel-yellow: #ffdd59;
|
||||||
$eve-solar-system-status-lookingFor: rgba(67, 176, 253, 0.48);
|
--dark-bg: #2d2d2d;
|
||||||
$eve-solar-system-status-home: rgb(197, 253, 67);
|
--text-color: #ffffff;
|
||||||
|
--tooltip-bg: #202020;
|
||||||
|
|
||||||
$eve-solar-system-status-color-unknown: transparent;
|
--pastel-blue-darken10: #4f6b86;
|
||||||
$eve-solar-system-status-color-friendly: #3bbd39;
|
--pastel-blue-lighten10: #6da3af;
|
||||||
$eve-solar-system-status-color-warning: #ffb93b;
|
--pastel-pink-darken10: #bb7ca9;
|
||||||
$eve-solar-system-status-color-target: #b439ff;
|
--pastel-pink-lighten10: #e0a6cb;
|
||||||
$eve-solar-system-status-color-dangerous: #d54040;
|
--pastel-green-darken10: #79a244;
|
||||||
$eve-solar-system-status-color-lookingFor: #43c2fd;
|
--pastel-green-lighten10: #99cf52;
|
||||||
$eve-solar-system-status-color-home: rgb(197, 253, 67);
|
--pastel-yellow-darken10: #e6c44f;
|
||||||
|
--pastel-yellow-lighten10: #ffe874;
|
||||||
|
|
||||||
|
--eve-link-color-default: #333;
|
||||||
|
--eve-link-color-top-mass-0: #333;
|
||||||
|
--eve-link-color-top-mass-1: #5a4520;
|
||||||
|
--eve-link-color-top-mass-2: #672c2c;
|
||||||
|
--eve-link-color-middle-mass-0: #333;
|
||||||
|
--eve-link-color-middle-mass-1: #333;
|
||||||
|
--eve-link-color-middle-mass-2: #333;
|
||||||
|
--eve-link-color-middle-time-0: #5c5c5c;
|
||||||
|
--eve-link-color-middle-time-1: #ff00cd;
|
||||||
|
--eve-link-color-middle-time-1-border: #99f3ff;
|
||||||
|
--eve-link-color-top-mass-1-time-1: #796300;
|
||||||
|
--eve-link-color-top-mass-2-time-1: #8c1717;
|
||||||
|
--eve-link-color-temp: orange;
|
||||||
|
|
||||||
|
--eve-effect-pulsar: #40aef5;
|
||||||
|
--eve-effect-magnetar: #f058f8;
|
||||||
|
--eve-effect-wolfRayet: #ef7843;
|
||||||
|
--eve-effect-blackHole: #1b1b1b;
|
||||||
|
--eve-effect-cataclysmicVariable: #ffea90;
|
||||||
|
--eve-effect-redGiant: #fd3c3c;
|
||||||
|
--eve-effect-dazhLiminalityLocus: #ff6464;
|
||||||
|
--eve-effect-imperialStellarObservatory: #6991ce;
|
||||||
|
--eve-effect-stateStellarObservatory: #6991ce;
|
||||||
|
--eve-effect-republicStellarObservatory: #6991ce;
|
||||||
|
--eve-effect-federalStellarObservatory: #6991ce;
|
||||||
|
|
||||||
|
--eve-wh-type-color-high: #5dffd2;
|
||||||
|
--eve-wh-type-color-low: #f79400;
|
||||||
|
--eve-wh-type-color-null: #fc3c3c;
|
||||||
|
--eve-wh-type-color-c1: #69bfce;
|
||||||
|
--eve-wh-type-color-c2: #6991ce;
|
||||||
|
--eve-wh-type-color-c3: #a8cb70;
|
||||||
|
--eve-wh-type-color-c4: #e39c68;
|
||||||
|
--eve-wh-type-color-c5: #de8686;
|
||||||
|
--eve-wh-type-color-c6: #e76363;
|
||||||
|
--eve-wh-type-color-c13: #988cb5;
|
||||||
|
--eve-wh-type-color-drifter: #ff44f6;
|
||||||
|
--eve-wh-type-color-thera: #ffffff;
|
||||||
|
--eve-wh-type-color-zarzakh: #212121;
|
||||||
|
|
||||||
|
--eve-security-color-10: #2c74df;
|
||||||
|
--eve-security-color-09: #3998e8;
|
||||||
|
--eve-security-color-08: #4dcbf5;
|
||||||
|
--eve-security-color-07: #60d8a2;
|
||||||
|
--eve-security-color-06: #71e454;
|
||||||
|
--eve-security-color-05: #f2fc81;
|
||||||
|
--eve-security-color-04: #d96c07;
|
||||||
|
--eve-security-color-03: #cb440f;
|
||||||
|
--eve-security-color-02: #b91117;
|
||||||
|
--eve-security-color-01: #732020;
|
||||||
|
--eve-security-color-00: #8b3263;
|
||||||
|
--eve-security-color-m-01: #8b3263;
|
||||||
|
--eve-security-color-m-02: #8b3263;
|
||||||
|
--eve-security-color-m-03: #8b3263;
|
||||||
|
--eve-security-color-m-04: #8b3263;
|
||||||
|
--eve-security-color-m-05: #8b3263;
|
||||||
|
--eve-security-color-m-06: #8b3263;
|
||||||
|
--eve-security-color-m-07: #8b3263;
|
||||||
|
--eve-security-color-m-08: #8b3263;
|
||||||
|
--eve-security-color-m-09: #8b3263;
|
||||||
|
--eve-security-color-m-10: #8b3263;
|
||||||
|
|
||||||
|
--eve-solar-system-status-unknown: transparent;
|
||||||
|
--eve-solar-system-status-color-unknown: transparent;
|
||||||
|
--eve-solar-system-status-home: #{$homeAlpha};
|
||||||
|
--eve-solar-system-status-color-home: #{$homeBase};
|
||||||
|
--eve-solar-system-status-color-home-dark30: #{$homeDark30};
|
||||||
|
--eve-solar-system-status-friendly: #{$friendlyAlpha};
|
||||||
|
--eve-solar-system-status-color-friendly: #{$friendlyBase};
|
||||||
|
--eve-solar-system-status-friendly-dark30: #{$friendlyDark30};
|
||||||
|
--eve-solar-system-status-color-friendly-dark20: #{$friendlyDark20};
|
||||||
|
--eve-solar-system-status-color-friendly-dark5: #{$friendlyDark5};
|
||||||
|
--eve-solar-system-status-lookingFor: #{$lookingForAlpha};
|
||||||
|
--eve-solar-system-status-color-lookingFor: #{$lookingForBase};
|
||||||
|
--eve-solar-system-status-color-lookingFor-dark15: #{$lookingForDark15};
|
||||||
|
--eve-solar-system-status-warning: #906518a6;
|
||||||
|
--eve-solar-system-status-color-warning: #ffb93b;
|
||||||
|
--eve-solar-system-status-target: #b439ff6b;
|
||||||
|
--eve-solar-system-status-color-target: #b439ff;
|
||||||
|
--eve-solar-system-status-dangerous: #d54040;
|
||||||
|
--eve-solar-system-status-color-dangerous: #d54040;
|
||||||
|
|
||||||
|
--conn-time-eol: #7452c3e3;
|
||||||
|
--conn-frigate: #325d88;
|
||||||
|
--conn-save: rgba(155, 102, 45, 0.85);
|
||||||
|
--selected-item-bg: rgba(98, 98, 98, 0.33);
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,535 +1,504 @@
|
|||||||
@import "eve-common-variables";
|
@import './eve-common-variables';
|
||||||
|
|
||||||
|
|
||||||
.eve-wh-effect-color-pulsar {
|
.eve-wh-effect-color-pulsar {
|
||||||
fill: $eve-effect-pulsar;
|
fill: var(--eve-effect-pulsar);
|
||||||
background-color: $eve-effect-pulsar;
|
background-color: var(--eve-effect-pulsar);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-effect-color-magnetar {
|
.eve-wh-effect-color-magnetar {
|
||||||
fill: $eve-effect-magnetar;
|
fill: var(--eve-effect-magnetar);
|
||||||
background-color: $eve-effect-magnetar;
|
background-color: var(--eve-effect-magnetar);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-effect-color-wolfRayet {
|
.eve-wh-effect-color-wolfRayet {
|
||||||
fill: $eve-effect-wolfRayet;
|
fill: var(--eve-effect-wolfRayet);
|
||||||
background-color: $eve-effect-wolfRayet;
|
background-color: var(--eve-effect-wolfRayet);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-effect-color-blackHole {
|
.eve-wh-effect-color-blackHole {
|
||||||
fill: $eve-effect-blackHole;
|
fill: var(--eve-effect-blackHole);
|
||||||
background-color: $eve-effect-blackHole;
|
background-color: var(--eve-effect-blackHole);
|
||||||
box-shadow: 0 0 8px rgba(255 255 255 / 33);
|
box-shadow: 0 0 8px rgba(255, 255, 255, 0.33);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-effect-color-cataclysmicVariable {
|
.eve-wh-effect-color-cataclysmicVariable {
|
||||||
fill: $eve-effect-cataclysmicVariable;
|
fill: var(--eve-effect-cataclysmicVariable);
|
||||||
background-color: $eve-effect-cataclysmicVariable;
|
background-color: var(--eve-effect-cataclysmicVariable);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-effect-color-redGiant {
|
.eve-wh-effect-color-redGiant {
|
||||||
fill: $eve-effect-redGiant;
|
fill: var(--eve-effect-redGiant);
|
||||||
background-color: $eve-effect-redGiant;
|
background-color: var(--eve-effect-redGiant);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-pulsar {
|
.text-eve-wh-effect-color-pulsar {
|
||||||
color: $eve-effect-pulsar;
|
color: var(--eve-effect-pulsar);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-magnetar {
|
.text-eve-wh-effect-color-magnetar {
|
||||||
color: $eve-effect-magnetar;
|
color: var(--eve-effect-magnetar);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-wolfRayet {
|
.text-eve-wh-effect-color-wolfRayet {
|
||||||
color: $eve-effect-wolfRayet;
|
color: var(--eve-effect-wolfRayet);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-blackHole {
|
.text-eve-wh-effect-color-blackHole {
|
||||||
color: #fff;
|
color: #fff;
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-cataclysmicVariable {
|
.text-eve-wh-effect-color-cataclysmicVariable {
|
||||||
color: $eve-effect-cataclysmicVariable;
|
color: var(--eve-effect-cataclysmicVariable);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-redGiant {
|
.text-eve-wh-effect-color-redGiant {
|
||||||
color: $eve-effect-redGiant;
|
color: var(--eve-effect-redGiant);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-dazhLiminalityLocus {
|
.text-eve-wh-effect-color-dazhLiminalityLocus {
|
||||||
color: $eve-effect-dazhLiminalityLocus;
|
color: var(--eve-effect-dazhLiminalityLocus);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-imperialStellarObservatory {
|
.text-eve-wh-effect-color-imperialStellarObservatory {
|
||||||
color: $eve-effect-imperialStellarObservatory;
|
color: var(--eve-effect-imperialStellarObservatory);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-stateStellarObservatory {
|
.text-eve-wh-effect-color-stateStellarObservatory {
|
||||||
color: $eve-effect-stateStellarObservatory;
|
color: var(--eve-effect-stateStellarObservatory);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-republicStellarObservatory {
|
.text-eve-wh-effect-color-republicStellarObservatory {
|
||||||
color: $eve-effect-republicStellarObservatory;
|
color: var(--eve-effect-republicStellarObservatory);
|
||||||
}
|
}
|
||||||
|
|
||||||
.text-eve-wh-effect-color-federalStellarObservatory {
|
.text-eve-wh-effect-color-federalStellarObservatory {
|
||||||
color: $eve-effect-federalStellarObservatory;
|
color: var(--eve-effect-federalStellarObservatory);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Security color classes */
|
||||||
.eve-security-color-10 {
|
.eve-security-color-10 {
|
||||||
color: $eve-security-color-10 !important;
|
color: var(--eve-security-color-10) !important;
|
||||||
fill: $eve-security-color-10;
|
fill: var(--eve-security-color-10);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-09 {
|
.eve-security-color-09 {
|
||||||
color: $eve-security-color-09 !important;
|
color: var(--eve-security-color-09) !important;
|
||||||
fill: $eve-security-color-09;
|
fill: var(--eve-security-color-09);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-08 {
|
.eve-security-color-08 {
|
||||||
color: $eve-security-color-08 !important;
|
color: var(--eve-security-color-08) !important;
|
||||||
fill: $eve-security-color-08;
|
fill: var(--eve-security-color-08);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-07 {
|
.eve-security-color-07 {
|
||||||
color: $eve-security-color-07 !important;
|
color: var(--eve-security-color-07) !important;
|
||||||
fill: $eve-security-color-07;
|
fill: var(--eve-security-color-07);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-06 {
|
.eve-security-color-06 {
|
||||||
color: $eve-security-color-06 !important;
|
color: var(--eve-security-color-06) !important;
|
||||||
fill: $eve-security-color-06;
|
fill: var(--eve-security-color-06);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-05 {
|
.eve-security-color-05 {
|
||||||
color: $eve-security-color-05 !important;
|
color: var(--eve-security-color-05) !important;
|
||||||
fill: $eve-security-color-05;
|
fill: var(--eve-security-color-05);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-04 {
|
.eve-security-color-04 {
|
||||||
color: $eve-security-color-04 !important;
|
color: var(--eve-security-color-04) !important;
|
||||||
fill: $eve-security-color-04;
|
fill: var(--eve-security-color-04);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-03 {
|
.eve-security-color-03 {
|
||||||
color: $eve-security-color-03 !important;
|
color: var(--eve-security-color-03) !important;
|
||||||
fill: $eve-security-color-03;
|
fill: var(--eve-security-color-03);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-02 {
|
.eve-security-color-02 {
|
||||||
color: $eve-security-color-02 !important;
|
color: var(--eve-security-color-02) !important;
|
||||||
fill: $eve-security-color-02;
|
fill: var(--eve-security-color-02);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-01 {
|
.eve-security-color-01 {
|
||||||
color: $eve-security-color-01 !important;
|
color: var(--eve-security-color-01) !important;
|
||||||
fill: $eve-security-color-01;
|
fill: var(--eve-security-color-01);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-00 {
|
.eve-security-color-00 {
|
||||||
color: $eve-security-color-00 !important;
|
color: var(--eve-security-color-00) !important;
|
||||||
fill: $eve-security-color-00;
|
fill: var(--eve-security-color-00);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-01 {
|
.eve-security-color-m-01 {
|
||||||
color: $eve-security-color-m-01 !important;
|
color: var(--eve-security-color-m-01) !important;
|
||||||
fill: $eve-security-color-m-01;
|
fill: var(--eve-security-color-m-01);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-02 {
|
.eve-security-color-m-02 {
|
||||||
color: $eve-security-color-m-02 !important;
|
color: var(--eve-security-color-m-02) !important;
|
||||||
fill: $eve-security-color-m-02;
|
fill: var(--eve-security-color-m-02);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-03 {
|
.eve-security-color-m-03 {
|
||||||
color: $eve-security-color-m-03 !important;
|
color: var(--eve-security-color-m-03) !important;
|
||||||
fill: $eve-security-color-m-03;
|
fill: var(--eve-security-color-m-03);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-04 {
|
.eve-security-color-m-04 {
|
||||||
color: $eve-security-color-m-04 !important;
|
color: var(--eve-security-color-m-04) !important;
|
||||||
fill: $eve-security-color-m-04;
|
fill: var(--eve-security-color-m-04);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-05 {
|
.eve-security-color-m-05 {
|
||||||
color: $eve-security-color-m-05 !important;
|
color: var(--eve-security-color-m-05) !important;
|
||||||
fill: $eve-security-color-m-05;
|
fill: var(--eve-security-color-m-05);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-06 {
|
.eve-security-color-m-06 {
|
||||||
color: $eve-security-color-m-06 !important;
|
color: var(--eve-security-color-m-06) !important;
|
||||||
fill: $eve-security-color-m-06;
|
fill: var(--eve-security-color-m-06);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-07 {
|
.eve-security-color-m-07 {
|
||||||
color: $eve-security-color-m-07 !important;
|
color: var(--eve-security-color-m-07) !important;
|
||||||
fill: $eve-security-color-m-07;
|
fill: var(--eve-security-color-m-07);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-08 {
|
.eve-security-color-m-08 {
|
||||||
color: $eve-security-color-m-08 !important;
|
color: var(--eve-security-color-m-08) !important;
|
||||||
fill: $eve-security-color-m-08;
|
fill: var(--eve-security-color-m-08);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-09 {
|
.eve-security-color-m-09 {
|
||||||
color: $eve-security-color-m-09 !important;
|
color: var(--eve-security-color-m-09) !important;
|
||||||
fill: $eve-security-color-m-09;
|
fill: var(--eve-security-color-m-09);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-color-m-10 {
|
.eve-security-color-m-10 {
|
||||||
color: $eve-security-color-m-10 !important;
|
color: var(--eve-security-color-m-10) !important;
|
||||||
fill: $eve-security-color-m-10;
|
fill: var(--eve-security-color-m-10);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Security backgrounds */
|
||||||
.eve-security-background-10 {
|
.eve-security-background-10 {
|
||||||
background-color: $eve-security-color-10;
|
background-color: var(--eve-security-color-10);
|
||||||
fill: $eve-security-color-10;
|
fill: var(--eve-security-color-10);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-09 {
|
.eve-security-background-09 {
|
||||||
background-color: $eve-security-color-09;
|
background-color: var(--eve-security-color-09);
|
||||||
fill: $eve-security-color-09;
|
fill: var(--eve-security-color-09);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-08 {
|
.eve-security-background-08 {
|
||||||
background-color: $eve-security-color-08;
|
background-color: var(--eve-security-color-08);
|
||||||
fill: $eve-security-color-08;
|
fill: var(--eve-security-color-08);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-07 {
|
.eve-security-background-07 {
|
||||||
background-color: $eve-security-color-07;
|
background-color: var(--eve-security-color-07);
|
||||||
fill: $eve-security-color-07;
|
fill: var(--eve-security-color-07);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-06 {
|
.eve-security-background-06 {
|
||||||
background-color: $eve-security-color-06;
|
background-color: var(--eve-security-color-06);
|
||||||
fill: $eve-security-color-06;
|
fill: var(--eve-security-color-06);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-05 {
|
.eve-security-background-05 {
|
||||||
background-color: $eve-security-color-05;
|
background-color: var(--eve-security-color-05);
|
||||||
fill: $eve-security-color-05;
|
fill: var(--eve-security-color-05);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-04 {
|
.eve-security-background-04 {
|
||||||
background-color: $eve-security-color-04;
|
background-color: var(--eve-security-color-04);
|
||||||
fill: $eve-security-color-04;
|
fill: var(--eve-security-color-04);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-03 {
|
.eve-security-background-03 {
|
||||||
background-color: $eve-security-color-03;
|
background-color: var(--eve-security-color-03);
|
||||||
fill: $eve-security-color-03;
|
fill: var(--eve-security-color-03);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-02 {
|
.eve-security-background-02 {
|
||||||
background-color: $eve-security-color-02;
|
background-color: var(--eve-security-color-02);
|
||||||
fill: $eve-security-color-02;
|
fill: var(--eve-security-color-02);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-01 {
|
.eve-security-background-01 {
|
||||||
background-color: $eve-security-color-01;
|
background-color: var(--eve-security-color-01);
|
||||||
fill: $eve-security-color-01;
|
fill: var(--eve-security-color-01);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-00 {
|
.eve-security-background-00 {
|
||||||
background-color: $eve-security-color-00;
|
background-color: var(--eve-security-color-00);
|
||||||
fill: $eve-security-color-00;
|
fill: var(--eve-security-color-00);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-01 {
|
.eve-security-background-m-01 {
|
||||||
background-color: $eve-security-color-m-01;
|
background-color: var(--eve-security-color-m-01);
|
||||||
fill: $eve-security-color-m-01;
|
fill: var(--eve-security-color-m-01);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-02 {
|
.eve-security-background-m-02 {
|
||||||
background-color: $eve-security-color-m-02;
|
background-color: var(--eve-security-color-m-02);
|
||||||
fill: $eve-security-color-m-02;
|
fill: var(--eve-security-color-m-02);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-03 {
|
.eve-security-background-m-03 {
|
||||||
background-color: $eve-security-color-m-03;
|
background-color: var(--eve-security-color-m-03);
|
||||||
fill: $eve-security-color-m-03;
|
fill: var(--eve-security-color-m-03);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-04 {
|
.eve-security-background-m-04 {
|
||||||
background-color: $eve-security-color-m-04;
|
background-color: var(--eve-security-color-m-04);
|
||||||
fill: $eve-security-color-m-04;
|
fill: var(--eve-security-color-m-04);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-05 {
|
.eve-security-background-m-05 {
|
||||||
background-color: $eve-security-color-m-05;
|
background-color: var(--eve-security-color-m-05);
|
||||||
fill: $eve-security-color-m-05;
|
fill: var(--eve-security-color-m-05);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-06 {
|
.eve-security-background-m-06 {
|
||||||
background-color: $eve-security-color-m-06;
|
background-color: var(--eve-security-color-m-06);
|
||||||
fill: $eve-security-color-m-06;
|
fill: var(--eve-security-color-m-06);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-07 {
|
.eve-security-background-m-07 {
|
||||||
background-color: $eve-security-color-m-07;
|
background-color: var(--eve-security-color-m-07);
|
||||||
fill: $eve-security-color-m-07;
|
fill: var(--eve-security-color-m-07);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-08 {
|
.eve-security-background-m-08 {
|
||||||
background-color: $eve-security-color-m-08;
|
background-color: var(--eve-security-color-m-08);
|
||||||
fill: $eve-security-color-m-08;
|
fill: var(--eve-security-color-m-08);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-09 {
|
.eve-security-background-m-09 {
|
||||||
background-color: $eve-security-color-m-09;
|
background-color: var(--eve-security-color-m-09);
|
||||||
fill: $eve-security-color-m-09;
|
fill: var(--eve-security-color-m-09);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-security-background-m-10 {
|
.eve-security-background-m-10 {
|
||||||
background-color: $eve-security-color-m-10;
|
background-color: var(--eve-security-color-m-10);
|
||||||
fill: $eve-security-color-m-10;
|
fill: var(--eve-security-color-m-10);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* WH Type color classes */
|
||||||
.eve-wh-type-color-high {
|
.eve-wh-type-color-high {
|
||||||
color: $eve-wh-type-color-high;
|
color: var(--eve-wh-type-color-high) !important;
|
||||||
fill: $eve-wh-type-color-high;
|
fill: var(--eve-wh-type-color-high);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-low {
|
.eve-wh-type-color-low {
|
||||||
color: $eve-wh-type-color-low;
|
color: var(--eve-wh-type-color-low) !important;
|
||||||
fill: $eve-wh-type-color-low;
|
fill: var(--eve-wh-type-color-low);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-null {
|
.eve-wh-type-color-null {
|
||||||
color: $eve-wh-type-color-null;
|
color: var(--eve-wh-type-color-null) !important;
|
||||||
fill: $eve-wh-type-color-null;
|
fill: var(--eve-wh-type-color-null);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-c1 {
|
.eve-wh-type-color-c1 {
|
||||||
color: $eve-wh-type-color-c1 !important;
|
color: var(--eve-wh-type-color-c1) !important;
|
||||||
fill: $eve-wh-type-color-c1;
|
fill: var(--eve-wh-type-color-c1);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-c2 {
|
.eve-wh-type-color-c2 {
|
||||||
color: $eve-wh-type-color-c2 !important;
|
color: var(--eve-wh-type-color-c2) !important;
|
||||||
fill: $eve-wh-type-color-c2;
|
fill: var(--eve-wh-type-color-c2);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-c3 {
|
.eve-wh-type-color-c3 {
|
||||||
color: $eve-wh-type-color-c3 !important;
|
color: var(--eve-wh-type-color-c3) !important;
|
||||||
fill: $eve-wh-type-color-c3;
|
fill: var(--eve-wh-type-color-c3);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-c4 {
|
.eve-wh-type-color-c4 {
|
||||||
color: $eve-wh-type-color-c4 !important;
|
color: var(--eve-wh-type-color-c4) !important;
|
||||||
fill: $eve-wh-type-color-c4;
|
fill: var(--eve-wh-type-color-c4);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-c5 {
|
.eve-wh-type-color-c5 {
|
||||||
color: $eve-wh-type-color-c5 !important;
|
color: var(--eve-wh-type-color-c5) !important;
|
||||||
fill: $eve-wh-type-color-c5;
|
fill: var(--eve-wh-type-color-c5);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-c6 {
|
.eve-wh-type-color-c6 {
|
||||||
color: $eve-wh-type-color-c6 !important;
|
color: var(--eve-wh-type-color-c6) !important;
|
||||||
fill: $eve-wh-type-color-c6;
|
fill: var(--eve-wh-type-color-c6);
|
||||||
font-weight: bold !important;
|
font-weight: bold !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-c13 {
|
.eve-wh-type-color-c13 {
|
||||||
color: $eve-wh-type-color-c13 !important;
|
color: var(--eve-wh-type-color-c13) !important;
|
||||||
fill: $eve-wh-type-color-c13;
|
fill: var(--eve-wh-type-color-c13);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-drifter {
|
.eve-wh-type-color-drifter {
|
||||||
color: $eve-wh-type-color-drifter !important;
|
color: var(--eve-wh-type-color-drifter) !important;
|
||||||
fill: $eve-wh-type-color-drifter;
|
fill: var(--eve-wh-type-color-drifter);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-color-thera {
|
.eve-wh-type-color-thera {
|
||||||
color: $eve-wh-type-color-thera !important;
|
color: var(--eve-wh-type-color-thera) !important;
|
||||||
fill: $eve-wh-type-color-thera;
|
fill: var(--eve-wh-type-color-thera);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* WH Type backgrounds */
|
||||||
.eve-wh-type-background-high {
|
.eve-wh-type-background-high {
|
||||||
background-color: $eve-wh-type-color-high;
|
background-color: var(--eve-wh-type-color-high);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-low {
|
.eve-wh-type-background-low {
|
||||||
background-color: $eve-wh-type-color-low;
|
background-color: var(--eve-wh-type-color-low);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-null {
|
.eve-wh-type-background-null {
|
||||||
background-color: $eve-wh-type-color-null;
|
background-color: var(--eve-wh-type-color-null);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-c1 {
|
.eve-wh-type-background-c1 {
|
||||||
background-color: $eve-wh-type-color-c1;
|
background-color: var(--eve-wh-type-color-c1);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-c2 {
|
.eve-wh-type-background-c2 {
|
||||||
background-color: $eve-wh-type-color-c2;
|
background-color: var(--eve-wh-type-color-c2);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-c3 {
|
.eve-wh-type-background-c3 {
|
||||||
background-color: $eve-wh-type-color-c3;
|
background-color: var(--eve-wh-type-color-c3);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-c4 {
|
.eve-wh-type-background-c4 {
|
||||||
background-color: $eve-wh-type-color-c4;
|
background-color: var(--eve-wh-type-color-c4);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-c5 {
|
.eve-wh-type-background-c5 {
|
||||||
background-color: $eve-wh-type-color-c5;
|
background-color: var(--eve-wh-type-color-c5);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-c6 {
|
.eve-wh-type-background-c6 {
|
||||||
background-color: $eve-wh-type-color-c6;
|
background-color: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-c13 {
|
.eve-wh-type-background-c13 {
|
||||||
background-color: $eve-wh-type-color-c13;
|
background-color: var(--eve-wh-type-color-c13);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-drifter {
|
.eve-wh-type-background-drifter {
|
||||||
background-color: $eve-wh-type-color-drifter;
|
background-color: var(--eve-wh-type-color-drifter);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-thera {
|
.eve-wh-type-background-thera {
|
||||||
background-color: $eve-wh-type-color-thera;
|
background-color: var(--eve-wh-type-color-thera);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-wh-type-background-zarzakh {
|
.eve-wh-type-background-zarzakh {
|
||||||
background-color: $eve-wh-type-color-zarzakh;
|
background-color: var(--eve-wh-type-color-zarzakh);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Kind color classes */
|
||||||
.eve-kind-color-high {
|
.eve-kind-color-high {
|
||||||
color: $eve-wh-type-color-high;
|
color: var(--eve-wh-type-color-high);
|
||||||
fill: $eve-wh-type-color-high;
|
fill: var(--eve-wh-type-color-high);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-color-low {
|
.eve-kind-color-low {
|
||||||
color: $eve-wh-type-color-low;
|
color: var(--eve-wh-type-color-low);
|
||||||
fill: $eve-wh-type-color-low;
|
fill: var(--eve-wh-type-color-low);
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-color-null {
|
.eve-kind-color-null {
|
||||||
color: $eve-wh-type-color-null;
|
color: var(--eve-wh-type-color-null);
|
||||||
fill: $eve-wh-type-color-null;
|
fill: var(--eve-wh-type-color-null);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-color-wh {
|
.eve-kind-color-wh {
|
||||||
color: $eve-wh-type-color-c6;
|
color: var(--eve-wh-type-color-c6);
|
||||||
fill: $eve-wh-type-color-c6;
|
fill: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-color-thera {
|
.eve-kind-color-thera {
|
||||||
color: $eve-wh-type-color-thera;
|
color: var(--eve-wh-type-color-thera);
|
||||||
fill: $eve-wh-type-color-thera;
|
fill: var(--eve-wh-type-color-thera);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-color-abyss {
|
.eve-kind-color-abyss {
|
||||||
color: $eve-wh-type-color-c6;
|
color: var(--eve-wh-type-color-c6);
|
||||||
fill: $eve-wh-type-color-c6;
|
fill: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-color-penalty {
|
.eve-kind-color-penalty {
|
||||||
color: $eve-wh-type-color-c6;
|
color: var(--eve-wh-type-color-c6);
|
||||||
fill: $eve-wh-type-color-c6;
|
fill: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-color-pochven {
|
.eve-kind-color-pochven {
|
||||||
color: $eve-wh-type-color-c6;
|
color: var(--eve-wh-type-color-c6);
|
||||||
fill: $eve-wh-type-color-c6;
|
fill: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-color-zarzakh {
|
.eve-kind-color-zarzakh {
|
||||||
color: $eve-wh-type-color-zarzakh;
|
color: var(--eve-wh-type-color-zarzakh);
|
||||||
fill: $eve-wh-type-color-zarzakh;
|
fill: var(--eve-wh-type-color-zarzakh);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Kind backgrounds */
|
||||||
.eve-kind-background-high {
|
.eve-kind-background-high {
|
||||||
background-color: $eve-wh-type-color-high;
|
background-color: var(--eve-wh-type-color-high);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-background-low {
|
.eve-kind-background-low {
|
||||||
background-color: $eve-wh-type-color-low;
|
background-color: var(--eve-wh-type-color-low);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-background-null {
|
.eve-kind-background-null {
|
||||||
background-color: $eve-wh-type-color-null;
|
background-color: var(--eve-wh-type-color-null);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-background-wh {
|
.eve-kind-background-wh {
|
||||||
background-color: $eve-wh-type-color-c6;
|
background-color: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-background-thera {
|
.eve-kind-background-thera {
|
||||||
background-color: $eve-wh-type-color-thera;
|
background-color: var(--eve-wh-type-color-thera);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-background-abyss {
|
.eve-kind-background-abyss {
|
||||||
background-color: $eve-wh-type-color-c6;
|
background-color: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-background-penalty {
|
.eve-kind-background-penalty {
|
||||||
background-color: $eve-wh-type-color-c6;
|
background-color: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-background-pochven {
|
.eve-kind-background-pochven {
|
||||||
background-color: $eve-wh-type-color-c6;
|
background-color: var(--eve-wh-type-color-c6);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-kind-background-zarzakh {
|
.eve-kind-background-zarzakh {
|
||||||
background-color: $eve-wh-type-color-zarzakh;
|
background-color: var(--eve-wh-type-color-zarzakh);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* System status color classes */
|
||||||
.eve-system-status-color-clear {
|
.eve-system-status-color-clear {
|
||||||
color: $eve-solar-system-status-color-unknown;
|
color: var(--eve-solar-system-status-color-unknown);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-system-status-color-home {
|
.eve-system-status-color-home {
|
||||||
color: $eve-solar-system-status-color-home;
|
color: var(--eve-solar-system-status-color-home);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-system-status-color-friendly {
|
.eve-system-status-color-friendly {
|
||||||
color: $eve-solar-system-status-color-friendly;
|
color: var(--eve-solar-system-status-color-friendly);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-system-status-color-lookingFor {
|
.eve-system-status-color-lookingFor {
|
||||||
color: $eve-solar-system-status-color-lookingFor;
|
color: var(--eve-solar-system-status-color-lookingFor);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-system-status-color-warning {
|
.eve-system-status-color-warning {
|
||||||
color: $eve-solar-system-status-color-warning;
|
color: var(--eve-solar-system-status-color-warning);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-system-status-color-target {
|
.eve-system-status-color-target {
|
||||||
color: $eve-solar-system-status-color-target;
|
color: var(--eve-solar-system-status-color-target);
|
||||||
|
}
|
||||||
|
.eve-system-status-color-dangerous {
|
||||||
|
color: var(--eve-solar-system-status-color-dangerous);
|
||||||
}
|
}
|
||||||
|
|
||||||
.eve-system-status-color-dangerous {
|
.eve-system-status-clear {
|
||||||
color: $eve-solar-system-status-color-dangerous;
|
background-color: var(--eve-solar-system-status-unknown);
|
||||||
|
}
|
||||||
|
.eve-system-status-home {
|
||||||
|
background-color: var(--eve-solar-system-status-home);
|
||||||
|
}
|
||||||
|
.eve-system-status-friendly {
|
||||||
|
background-color: var(--eve-solar-system-status-friendly);
|
||||||
|
}
|
||||||
|
.eve-system-status-lookingFor {
|
||||||
|
background-color: var(--eve-solar-system-status-lookingFor);
|
||||||
|
}
|
||||||
|
.eve-system-status-warning {
|
||||||
|
background-color: var(--eve-solar-system-status-warning);
|
||||||
|
}
|
||||||
|
.eve-system-status-target {
|
||||||
|
background-color: var(--eve-solar-system-status-target);
|
||||||
|
}
|
||||||
|
.eve-system-status-dangerous {
|
||||||
|
background-color: var(--eve-solar-system-status-dangerous);
|
||||||
|
}
|
||||||
|
|
||||||
|
.eve-system-status-clear {
|
||||||
|
background-color: var(--eve-solar-system-status-unknown);
|
||||||
|
color: var(--eve-solar-system-status-color-unknown);
|
||||||
|
}
|
||||||
|
|
||||||
|
.eve-system-status-home {
|
||||||
|
background-color: var(--eve-solar-system-status-home);
|
||||||
|
color: var(--eve-solar-system-status-color-home);
|
||||||
|
}
|
||||||
|
|
||||||
|
.eve-system-status-friendly {
|
||||||
|
background-color: var(--eve-solar-system-status-friendly);
|
||||||
|
color: var(--eve-solar-system-status-color-friendly);
|
||||||
|
}
|
||||||
|
|
||||||
|
.eve-system-status-lookingFor {
|
||||||
|
background-color: var(--eve-solar-system-status-lookingFor);
|
||||||
|
color: var(--eve-solar-system-status-color-lookingFor);
|
||||||
|
}
|
||||||
|
|
||||||
|
.eve-system-status-warning {
|
||||||
|
background-color: var(--eve-solar-system-status-warning);
|
||||||
|
color: var(--eve-solar-system-status-color-warning);
|
||||||
|
}
|
||||||
|
|
||||||
|
.eve-system-status-target {
|
||||||
|
background-color: var(--eve-solar-system-status-target);
|
||||||
|
color: var(--eve-solar-system-status-color-target);
|
||||||
|
}
|
||||||
|
|
||||||
|
.eve-system-status-dangerous {
|
||||||
|
background-color: var(--eve-solar-system-status-dangerous);
|
||||||
|
color: var(--eve-solar-system-status-color-dangerous);
|
||||||
}
|
}
|
||||||
|
|
||||||
.wd-route-system-shape-triangle {
|
.wd-route-system-shape-triangle {
|
||||||
clip-path: polygon(50% 0, 0 100%, 100% 100%);
|
clip-path: polygon(50% 0, 0 100%, 100% 100%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.wd-route-system-shape-circle {
|
.wd-route-system-shape-circle {
|
||||||
border-radius: 40%;
|
border-radius: 40%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Some additional background classes */
|
||||||
.wd-marker-bookmark-color-shattered {
|
.wd-marker-bookmark-color-shattered {
|
||||||
background-color: #833ca4;
|
background-color: #833ca4;
|
||||||
margin-top: 1px;
|
margin-top: 1px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.wd-marker-bookmark-color-custom {
|
.wd-marker-bookmark-color-custom {
|
||||||
background-color: #282828;
|
background-color: #282828;
|
||||||
border: 1px solid #4c4c4c;
|
border: 1px solid #4c4c4c;
|
||||||
@@ -572,3 +541,49 @@
|
|||||||
.wd-marker-bookmark-color-danger {
|
.wd-marker-bookmark-color-danger {
|
||||||
background-color: #d10600;
|
background-color: #d10600;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.react-flow {
|
||||||
|
color: var(--text-color);
|
||||||
|
|
||||||
|
&__pane {
|
||||||
|
cursor: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__minimap {
|
||||||
|
background-color: rgba(66, 66, 66, 1);
|
||||||
|
opacity: 0.7;
|
||||||
|
border: 1px solid #2f2f2f;
|
||||||
|
border-radius: 4px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__minimap-mask {
|
||||||
|
fill: rgba(28, 28, 28, 0.75);
|
||||||
|
}
|
||||||
|
|
||||||
|
&__controls {
|
||||||
|
filter: brightness(1.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
&__minimap-node {
|
||||||
|
fill: #ffb03a;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.context-menu-active {
|
||||||
|
background-color: rgba(131, 131, 131, 0.33);
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dialog {
|
||||||
|
.p-dialog-header {
|
||||||
|
height: 40px;
|
||||||
|
padding: 1rem;
|
||||||
|
padding-right: 10px !important;
|
||||||
|
}
|
||||||
|
.p-dialog-title {
|
||||||
|
font-size: 1rem !important;
|
||||||
|
}
|
||||||
|
.p-dialog-header-icons {
|
||||||
|
align-self: initial !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
2
assets/js/hooks/Mapper/components/map/styles/index.scss
Normal file
2
assets/js/hooks/Mapper/components/map/styles/index.scss
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
@import './default-theme.scss';
|
||||||
|
@import './pathfinder-theme.scss';
|
||||||
@@ -1,74 +0,0 @@
|
|||||||
$pastel-blue: #5a7d9a;
|
|
||||||
$pastel-pink: #d291bc;
|
|
||||||
$pastel-green: #88b04b;
|
|
||||||
$pastel-yellow: #ffdd59;
|
|
||||||
$dark-bg: #2d2d2d;
|
|
||||||
$text-color: #ffffff;
|
|
||||||
$tooltip-bg: #202020;
|
|
||||||
|
|
||||||
.react-flow {
|
|
||||||
// background-color: $dark-bg;
|
|
||||||
color: $text-color;
|
|
||||||
|
|
||||||
&__node {
|
|
||||||
//cursor: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
&__pane {
|
|
||||||
cursor: auto;
|
|
||||||
}
|
|
||||||
//&__edge {
|
|
||||||
// stroke: $pastel-pink;
|
|
||||||
// stroke-width: 2px;
|
|
||||||
//
|
|
||||||
// &.selected {
|
|
||||||
// stroke: $pastel-yellow;
|
|
||||||
// }
|
|
||||||
//}
|
|
||||||
|
|
||||||
&__handle {
|
|
||||||
//background-color: $pastel-green;
|
|
||||||
//box-shadow: 0 0 5px rgba($pastel-green, 0.5);
|
|
||||||
}
|
|
||||||
|
|
||||||
&__minimap {
|
|
||||||
background-color: rgba(66, 66, 66, 1);
|
|
||||||
opacity: 0.7;
|
|
||||||
//backdrop-filter: blur(5px);
|
|
||||||
border: 1px solid #2f2f2f;
|
|
||||||
border-radius: 4px;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
&__minimap-mask {
|
|
||||||
fill: rgba(28, 28, 28, 0.75);
|
|
||||||
}
|
|
||||||
|
|
||||||
&__controls {
|
|
||||||
filter: brightness(1.5);
|
|
||||||
}
|
|
||||||
|
|
||||||
&__minimap-node {
|
|
||||||
fill: #ffb03a;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.context-menu-active {
|
|
||||||
background-color: rgba(131, 131, 131, 0.33);
|
|
||||||
}
|
|
||||||
|
|
||||||
.p-dialog {
|
|
||||||
.p-dialog-header {
|
|
||||||
height: 40px;
|
|
||||||
padding: 1rem;
|
|
||||||
padding-right: 10px !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.p-dialog-title {
|
|
||||||
font-size: 1rem !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.p-dialog-header-icons {
|
|
||||||
align-self: initial !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
$pastel-blue: #5a7d9a;
|
|
||||||
$pastel-pink: #d291bc;
|
|
||||||
$pastel-green: #88b04b;
|
|
||||||
$pastel-yellow: #ffdd59;
|
|
||||||
$dark-bg: #2d2d2d;
|
|
||||||
$text-color: #ffffff;
|
|
||||||
$tooltip-bg: #202020;
|
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
@import './eve-common-variables';
|
||||||
|
@import './eve-common';
|
||||||
|
@import url('https://fonts.googleapis.com/css2?family=Oxygen:wght@300;400;700&display=swap');
|
||||||
|
|
||||||
|
.pathfinder-theme {
|
||||||
|
--rf-bg-color: #000000;
|
||||||
|
--rf-soft-bg-color: #282828;
|
||||||
|
|
||||||
|
--rf-node-bg-color: #202020;
|
||||||
|
--rf-node-soft-bg-color: #313335;
|
||||||
|
--rf-node-font-weight: bold;
|
||||||
|
--rf-text-color: #adadad;
|
||||||
|
--rf-region-name: var(--rf-text-color);
|
||||||
|
--rf-custom-name: var(--rf-text-color);
|
||||||
|
|
||||||
|
--tooltip-bg: #202020;
|
||||||
|
|
||||||
|
--rf-bg-variant: "lines";
|
||||||
|
--rf-bg-gap: 32;
|
||||||
|
--rf-bg-size: 1;
|
||||||
|
--rf-bg-pattern-color: #313131;
|
||||||
|
|
||||||
|
--eve-effect-pulsar: #428bca;
|
||||||
|
--eve-effect-magnetar: #e06fdf;
|
||||||
|
--eve-effect-wolfRayet: #e28a0d;
|
||||||
|
--eve-effect-blackHole: #000000;
|
||||||
|
--eve-effect-cataclysmicVariable: #ffffbb;
|
||||||
|
--eve-effect-redGiant: #d9534f;
|
||||||
|
|
||||||
|
--eve-wh-type-color-high: #5cb85c;
|
||||||
|
--eve-wh-type-color-low: #e28a0d;
|
||||||
|
--eve-wh-type-color-null: #d9534f;
|
||||||
|
--eve-wh-type-color-c1: #428bca;
|
||||||
|
--eve-wh-type-color-c2: #428bca;
|
||||||
|
--eve-wh-type-color-c3: #e28a0d;
|
||||||
|
--eve-wh-type-color-c4: #e28a0d;
|
||||||
|
--eve-wh-type-color-c5: #d9534f;
|
||||||
|
--eve-wh-type-color-c6: #d9534f;
|
||||||
|
--eve-wh-type-color-c13: #7986cb;
|
||||||
|
--eve-wh-type-color-drifter: #44aa82;
|
||||||
|
|
||||||
|
--rf-node-font-weight: bold;
|
||||||
|
--rf-node-line-height: normal;
|
||||||
|
--rf-node-font-family: 'Oxygen', sans-serif;
|
||||||
|
--rf-node-text-color: var(--pf-text-color);
|
||||||
|
|
||||||
|
--rf-tag-color: #fbbf24;
|
||||||
|
--rf-has-user-characters: #5cb85c;
|
||||||
|
|
||||||
|
--window-corner: #72716f;
|
||||||
|
}
|
||||||
@@ -1,78 +1,25 @@
|
|||||||
import 'react-grid-layout/css/styles.css';
|
import 'react-grid-layout/css/styles.css';
|
||||||
import 'react-resizable/css/styles.css';
|
import 'react-resizable/css/styles.css';
|
||||||
import { WidgetGridItem, WidgetsGrid } from '@/hooks/Mapper/components/mapInterface/components';
|
import { useMemo } from 'react';
|
||||||
import {
|
import { WindowManager } from '@/hooks/Mapper/components/ui-kit/WindowManager';
|
||||||
LocalCharacters,
|
import { DEFAULT_WIDGETS } from '@/hooks/Mapper/components/mapInterface/constants.tsx';
|
||||||
RoutesWidget,
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
SystemInfo,
|
|
||||||
SystemSignatures,
|
|
||||||
} from '@/hooks/Mapper/components/mapInterface/widgets';
|
|
||||||
import { useState } from 'react';
|
|
||||||
import { SESSION_KEY } from '@/hooks/Mapper/constants.ts';
|
|
||||||
// import { debounce } from 'lodash/debounce';
|
|
||||||
|
|
||||||
const DEFAULT_WINDOWS = [
|
|
||||||
{
|
|
||||||
name: 'info',
|
|
||||||
rightOffset: 5,
|
|
||||||
width: 5,
|
|
||||||
height: 4,
|
|
||||||
item: () => <SystemInfo />,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
name: 'local',
|
|
||||||
rightOffset: 5,
|
|
||||||
topOffset: 4,
|
|
||||||
width: 5,
|
|
||||||
height: 4,
|
|
||||||
item: () => <LocalCharacters />,
|
|
||||||
},
|
|
||||||
{ name: 'signatures', width: 8, height: 4, topOffset: 8, rightOffset: 12, item: () => <SystemSignatures /> },
|
|
||||||
{
|
|
||||||
name: 'routes',
|
|
||||||
rightOffset: 0,
|
|
||||||
topOffset: 8,
|
|
||||||
width: 5,
|
|
||||||
height: 6,
|
|
||||||
item: () => <RoutesWidget />,
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
const saveWindowsToLS = (toSaveItems: WidgetGridItem[]) => {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
||||||
const out = toSaveItems.map(({ item, ...rest }) => rest);
|
|
||||||
localStorage.setItem(SESSION_KEY.windows, JSON.stringify(out));
|
|
||||||
};
|
|
||||||
|
|
||||||
const restoreWindowsFromLS = (): WidgetGridItem[] => {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
||||||
const raw = localStorage.getItem(SESSION_KEY.windows);
|
|
||||||
if (!raw) {
|
|
||||||
console.warn('No windows found in local storage!!');
|
|
||||||
return DEFAULT_WINDOWS;
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line no-debugger
|
|
||||||
const out = (JSON.parse(raw) as Omit<WidgetGridItem, 'item'>[])
|
|
||||||
.filter(x => DEFAULT_WINDOWS.find(def => def.name === x.name))
|
|
||||||
.map(x => {
|
|
||||||
const windowItem = DEFAULT_WINDOWS.find(def => def.name === x.name)?.item;
|
|
||||||
return { ...x, item: windowItem! };
|
|
||||||
});
|
|
||||||
|
|
||||||
return out;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const MapInterface = () => {
|
export const MapInterface = () => {
|
||||||
const [items, setItems] = useState<WidgetGridItem[]>(restoreWindowsFromLS);
|
// const [items, setItems] = useState<WindowProps[]>(restoreWindowsFromLS);
|
||||||
|
const { windowsSettings, updateWidgetSettings } = useMapRootState();
|
||||||
|
|
||||||
return (
|
const items = useMemo(() => {
|
||||||
<WidgetsGrid
|
return windowsSettings.windows
|
||||||
items={items}
|
.map(x => {
|
||||||
onChange={x => {
|
const content = DEFAULT_WIDGETS.find(y => y.id === x.id)?.content;
|
||||||
saveWindowsToLS(x);
|
return {
|
||||||
setItems(x);
|
...x,
|
||||||
}}
|
content: content!,
|
||||||
/>
|
};
|
||||||
);
|
})
|
||||||
|
.filter(x => windowsSettings.visible.some(j => x.id === j));
|
||||||
|
}, [windowsSettings]);
|
||||||
|
|
||||||
|
return <WindowManager windows={items} dragSelector=".react-grid-dragHandleExample" onChange={updateWidgetSettings} />;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useRef } from 'react';
|
import { useCallback, useEffect, useRef } from 'react';
|
||||||
import { Dialog } from 'primereact/dialog';
|
import { Dialog } from 'primereact/dialog';
|
||||||
|
|
||||||
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
@@ -58,7 +58,7 @@ export const SystemLinkSignatureDialog = ({ data, setVisible }: SystemLinkSignat
|
|||||||
<Dialog
|
<Dialog
|
||||||
header="Select signature to link"
|
header="Select signature to link"
|
||||||
visible
|
visible
|
||||||
draggable={false}
|
draggable={true}
|
||||||
style={{ width: '500px' }}
|
style={{ width: '500px' }}
|
||||||
onHide={handleHide}
|
onHide={handleHide}
|
||||||
contentClassName="!p-0"
|
contentClassName="!p-0"
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import { InputTextarea } from 'primereact/inputtextarea';
|
|||||||
import { Dialog } from 'primereact/dialog';
|
import { Dialog } from 'primereact/dialog';
|
||||||
import { getSystemById } from '@/hooks/Mapper/helpers';
|
import { getSystemById } from '@/hooks/Mapper/helpers';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
import { useCallback, useEffect, useRef, useState } from 'react';
|
import { useCallback, useEffect, useRef, useState } from 'react';
|
||||||
import { Button } from 'primereact/button';
|
import { Button } from 'primereact/button';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types';
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
@@ -22,30 +23,21 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
|
|||||||
outCommand,
|
outCommand,
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const isTempSystemNameEnabled = useMapGetOption('show_temp_system_name') === 'true';
|
||||||
|
|
||||||
const system = getSystemById(systems, systemId);
|
const system = getSystemById(systems, systemId);
|
||||||
|
|
||||||
const [name, setName] = useState('');
|
const [name, setName] = useState('');
|
||||||
const [label, setLabel] = useState('');
|
const [label, setLabel] = useState('');
|
||||||
|
const [temporaryName, setTemporaryName] = useState('');
|
||||||
const [description, setDescription] = useState('');
|
const [description, setDescription] = useState('');
|
||||||
const inputRef = useRef<HTMLInputElement>();
|
const inputRef = useRef<HTMLInputElement>();
|
||||||
|
|
||||||
useEffect(() => {
|
const ref = useRef({ name, description, temporaryName, label, outCommand, systemId, system });
|
||||||
if (!system) {
|
ref.current = { name, description, label, temporaryName, outCommand, systemId, system };
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const labels = new LabelsManager(system.labels || '');
|
|
||||||
|
|
||||||
setName(system.name || '');
|
|
||||||
setLabel(labels.customLabel);
|
|
||||||
setDescription(system.description || '');
|
|
||||||
}, [system]);
|
|
||||||
|
|
||||||
const ref = useRef({ name, description, label, outCommand, systemId, system });
|
|
||||||
ref.current = { name, description, label, outCommand, systemId, system };
|
|
||||||
|
|
||||||
const handleSave = useCallback(() => {
|
const handleSave = useCallback(() => {
|
||||||
const { name, description, label, outCommand, systemId, system } = ref.current;
|
const { name, description, label, temporaryName, outCommand, systemId, system } = ref.current;
|
||||||
|
|
||||||
const outLabel = new LabelsManager(system?.labels ?? '');
|
const outLabel = new LabelsManager(system?.labels ?? '');
|
||||||
outLabel.updateCustomLabel(label);
|
outLabel.updateCustomLabel(label);
|
||||||
@@ -58,6 +50,14 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
outCommand({
|
||||||
|
type: OutCommand.updateSystemTemporaryName,
|
||||||
|
data: {
|
||||||
|
system_id: systemId,
|
||||||
|
value: temporaryName,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
outCommand({
|
outCommand({
|
||||||
type: OutCommand.updateSystemName,
|
type: OutCommand.updateSystemName,
|
||||||
data: {
|
data: {
|
||||||
@@ -93,6 +93,21 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
|
|||||||
e.target.value = e.target.value.toUpperCase().replace(/[^A-Z0-9\-[\](){}]/g, '');
|
e.target.value = e.target.value.toUpperCase().replace(/[^A-Z0-9\-[\](){}]/g, '');
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
// Attention: this effect should be call only on mount.
|
||||||
|
useEffect(() => {
|
||||||
|
const { system } = ref.current;
|
||||||
|
if (!system) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const labels = new LabelsManager(system.labels || '');
|
||||||
|
|
||||||
|
setName(system.name || '');
|
||||||
|
setLabel(labels.customLabel);
|
||||||
|
setTemporaryName(system.temporary_name || '');
|
||||||
|
setDescription(system.description || '');
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
header="System settings"
|
header="System settings"
|
||||||
@@ -167,6 +182,35 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
|
|||||||
</IconField>
|
</IconField>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{isTempSystemNameEnabled && (
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<label htmlFor="username">Temporary Name</label>
|
||||||
|
|
||||||
|
<IconField>
|
||||||
|
{temporaryName !== '' && (
|
||||||
|
<WdImgButton
|
||||||
|
className="pi pi-trash text-red-400"
|
||||||
|
textSize={WdImageSize.large}
|
||||||
|
tooltip={{
|
||||||
|
content: 'Remove temporary name',
|
||||||
|
className: 'pi p-input-icon',
|
||||||
|
position: TooltipPosition.top,
|
||||||
|
}}
|
||||||
|
onClick={() => setTemporaryName('')}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<InputText
|
||||||
|
id="temporaryName"
|
||||||
|
aria-describedby="temporaryName"
|
||||||
|
autoComplete="off"
|
||||||
|
value={temporaryName}
|
||||||
|
maxLength={10}
|
||||||
|
onChange={e => setTemporaryName(e.target.value)}
|
||||||
|
/>
|
||||||
|
</IconField>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
<div className="flex flex-col gap-1">
|
<div className="flex flex-col gap-1">
|
||||||
<label htmlFor="username">Description</label>
|
<label htmlFor="username">Description</label>
|
||||||
<InputTextarea
|
<InputTextarea
|
||||||
|
|||||||
@@ -1,37 +0,0 @@
|
|||||||
.GridLayoutWrapper {
|
|
||||||
width: 100%;
|
|
||||||
height: 100% !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.GridLayout {
|
|
||||||
width: 100%;
|
|
||||||
height: 100% !important;
|
|
||||||
pointer-events: none;
|
|
||||||
|
|
||||||
& > div {
|
|
||||||
pointer-events: initial;
|
|
||||||
}
|
|
||||||
|
|
||||||
:global {
|
|
||||||
.react-resizable-handle::after {
|
|
||||||
border-color: #696969 !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.react-grid-placeholder {
|
|
||||||
background-color: rgba(147, 147, 147, 0.3);
|
|
||||||
//filter: blur(5px);
|
|
||||||
border: 2px dashed #b6b6b6;
|
|
||||||
}
|
|
||||||
|
|
||||||
.react-grid-item {
|
|
||||||
transition-property: none !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.react-grid-item.cssTransforms {
|
|
||||||
transition-property: none !important;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,196 +0,0 @@
|
|||||||
import React, { useEffect, useRef, useState } from 'react';
|
|
||||||
|
|
||||||
import classes from './WidgetsGrid.module.scss';
|
|
||||||
import { ItemCallback, Layouts, Responsive, WidthProvider } from 'react-grid-layout';
|
|
||||||
import clsx from 'clsx';
|
|
||||||
import usePageVisibility from '@/hooks/Mapper/hooks/usePageVisibility.ts';
|
|
||||||
|
|
||||||
const ResponsiveGridLayout = WidthProvider(Responsive);
|
|
||||||
|
|
||||||
const colSize = 50;
|
|
||||||
const initState = { breakpoints: 100, cols: 2 };
|
|
||||||
|
|
||||||
export type WidgetGridItem = {
|
|
||||||
rightOffset?: number;
|
|
||||||
leftOffset?: number;
|
|
||||||
topOffset?: number;
|
|
||||||
width: number;
|
|
||||||
height: number;
|
|
||||||
name: string;
|
|
||||||
item: () => React.ReactNode;
|
|
||||||
};
|
|
||||||
|
|
||||||
export interface WidgetsGridProps {
|
|
||||||
items: WidgetGridItem[];
|
|
||||||
onChange: (items: WidgetGridItem[]) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const WidgetsGrid = ({ items, onChange }: WidgetsGridProps) => {
|
|
||||||
const containerRef = useRef<HTMLDivElement>(null);
|
|
||||||
const [, setKey] = useState(0);
|
|
||||||
const [callRerenderOfGrid, setCallRerenderOfGrid] = useState(0);
|
|
||||||
|
|
||||||
const isTabVisible = usePageVisibility();
|
|
||||||
|
|
||||||
const refAll = useRef({
|
|
||||||
isReady: false,
|
|
||||||
layouts: {
|
|
||||||
lg: [
|
|
||||||
// { i: 'a', w: 4, h: 16, x: 22, y: 0 },
|
|
||||||
// { i: 'b', w: 5, h: 10, x: 17, y: 0 },
|
|
||||||
],
|
|
||||||
} as Layouts,
|
|
||||||
breakpoints: { lg: 100, md: 0, sm: 0, xs: 0, xxs: 0 },
|
|
||||||
cols: { lg: 26, md: 0, sm: 0, xs: 0, xxs: 0 },
|
|
||||||
containerWidth: 0,
|
|
||||||
colsPrev: 26,
|
|
||||||
needPostProcess: false,
|
|
||||||
items: [...items],
|
|
||||||
});
|
|
||||||
|
|
||||||
// TODO
|
|
||||||
// 1. onLayoutChange (original) not calling when we change x of any widget
|
|
||||||
// 2. setKey need no call rerender for update props
|
|
||||||
const onLayoutChange: ItemCallback = (newItems, _, newItem) => {
|
|
||||||
const updatedItems = newItems.map(item => {
|
|
||||||
const toLeft = (item.x + item.w / 2) / refAll.current.cols.lg <= 0.5;
|
|
||||||
const original = refAll.current.items.find(x => x.name === item.i)!;
|
|
||||||
|
|
||||||
return {
|
|
||||||
...original,
|
|
||||||
width: item.w,
|
|
||||||
height: item.h,
|
|
||||||
leftOffset: toLeft ? item.x : undefined,
|
|
||||||
rightOffset: !toLeft ? refAll.current.cols.lg - (item.x + item.w) : undefined,
|
|
||||||
topOffset: item.y,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const sortedItems = [
|
|
||||||
...updatedItems.filter(x => x.name !== newItem.i),
|
|
||||||
updatedItems.find(x => x.name === newItem.i)!,
|
|
||||||
];
|
|
||||||
|
|
||||||
refAll.current.layouts = {
|
|
||||||
lg: [...newItems.filter(x => x.i !== newItem.i), newItem],
|
|
||||||
};
|
|
||||||
|
|
||||||
onChange(sortedItems);
|
|
||||||
setKey(x => x + 1);
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
refAll.current.items = [...items];
|
|
||||||
setKey(x => x + 1);
|
|
||||||
}, [items]);
|
|
||||||
|
|
||||||
// TODO
|
|
||||||
// 1. Unknown why but if we set layout and cols both instantly it not help...
|
|
||||||
// 1.2 it means that we should make report... until we will send new key on window resize
|
|
||||||
useEffect(() => {
|
|
||||||
const updateItems = () => {
|
|
||||||
if (!containerRef.current) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { width } = containerRef.current.getBoundingClientRect();
|
|
||||||
const newColsCount = (width - (width % colSize)) / colSize;
|
|
||||||
|
|
||||||
refAll.current.layouts = {
|
|
||||||
lg: refAll.current.items.map(({ name, width, height, rightOffset, leftOffset, topOffset = 0 }) => {
|
|
||||||
return {
|
|
||||||
i: name,
|
|
||||||
x: rightOffset != null ? newColsCount - width - rightOffset : leftOffset ?? 0,
|
|
||||||
y: topOffset,
|
|
||||||
w: width,
|
|
||||||
h: height,
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
refAll.current.cols = { lg: newColsCount, md: 0, sm: 0, xs: 0, xxs: 0 };
|
|
||||||
};
|
|
||||||
|
|
||||||
const updateContainerWidth = () => {
|
|
||||||
if (!containerRef.current) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { width } = containerRef.current.getBoundingClientRect();
|
|
||||||
|
|
||||||
refAll.current.containerWidth = width;
|
|
||||||
const newColsCount = (width - (width % colSize)) / colSize;
|
|
||||||
|
|
||||||
if (width <= 100 || refAll.current.cols.lg === newColsCount) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!refAll.current.isReady) {
|
|
||||||
updateItems();
|
|
||||||
setCallRerenderOfGrid(x => x + 1);
|
|
||||||
refAll.current.isReady = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
refAll.current.layouts = {
|
|
||||||
lg: refAll.current.layouts.lg.map(lgEl => {
|
|
||||||
const toLeft = (lgEl.x + lgEl.w / 2) / refAll.current.cols.lg <= 0.5;
|
|
||||||
const next = {
|
|
||||||
...lgEl,
|
|
||||||
x: toLeft ? lgEl.x : newColsCount - (refAll.current.cols.lg - lgEl.x),
|
|
||||||
};
|
|
||||||
return next;
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
|
|
||||||
refAll.current.cols = { lg: newColsCount, md: 0, sm: 0, xs: 0, xxs: 0 };
|
|
||||||
setCallRerenderOfGrid(x => x + 1);
|
|
||||||
};
|
|
||||||
|
|
||||||
setTimeout(() => updateContainerWidth(), 100);
|
|
||||||
|
|
||||||
const withRerender = () => {
|
|
||||||
updateContainerWidth();
|
|
||||||
setCallRerenderOfGrid(x => x + 1);
|
|
||||||
};
|
|
||||||
|
|
||||||
window.addEventListener('resize', withRerender);
|
|
||||||
return () => {
|
|
||||||
window.removeEventListener('resize', withRerender);
|
|
||||||
};
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const isNotSet = initState.cols === refAll.current.cols.lg;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div ref={containerRef} className={clsx(classes.GridLayoutWrapper, 'relative p-4')}>
|
|
||||||
{!isNotSet && isTabVisible && (
|
|
||||||
<ResponsiveGridLayout
|
|
||||||
key={callRerenderOfGrid}
|
|
||||||
className={classes.GridLayout}
|
|
||||||
layouts={refAll.current.layouts}
|
|
||||||
breakpoints={refAll.current.breakpoints}
|
|
||||||
cols={refAll.current.cols}
|
|
||||||
rowHeight={30}
|
|
||||||
width={refAll.current.containerWidth}
|
|
||||||
preventCollision={true}
|
|
||||||
compactType={null}
|
|
||||||
allowOverlap
|
|
||||||
onDragStop={onLayoutChange}
|
|
||||||
onResizeStop={onLayoutChange}
|
|
||||||
// onResizeStart={onLayoutChange}
|
|
||||||
// onDragStart={onLayoutChange}
|
|
||||||
isBounded
|
|
||||||
containerPadding={[0, 0]}
|
|
||||||
resizeHandles={['sw', 'se']}
|
|
||||||
draggableHandle=".react-grid-dragHandleExample"
|
|
||||||
>
|
|
||||||
{refAll.current.items.map(x => (
|
|
||||||
<div key={x.name} className="grid-item">
|
|
||||||
{x.item()}
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
</ResponsiveGridLayout>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
export * from './WidgetsGrid';
|
|
||||||
@@ -1,5 +1,4 @@
|
|||||||
export * from './Widget';
|
export * from './Widget';
|
||||||
export * from './WidgetsGrid';
|
|
||||||
export * from './SystemSettingsDialog';
|
export * from './SystemSettingsDialog';
|
||||||
export * from './SystemCustomLabelDialog';
|
export * from './SystemCustomLabelDialog';
|
||||||
export * from './SystemLinkSignatureDialog';
|
export * from './SystemLinkSignatureDialog';
|
||||||
|
|||||||
115
assets/js/hooks/Mapper/components/mapInterface/constants.tsx
Normal file
115
assets/js/hooks/Mapper/components/mapInterface/constants.tsx
Normal file
@@ -0,0 +1,115 @@
|
|||||||
|
import { WindowProps } from '@/hooks/Mapper/components/ui-kit/WindowManager/types.ts';
|
||||||
|
import {
|
||||||
|
LocalCharacters,
|
||||||
|
RoutesWidget,
|
||||||
|
SystemInfo,
|
||||||
|
SystemSignatures,
|
||||||
|
SystemStructures,
|
||||||
|
SystemKills,
|
||||||
|
} from '@/hooks/Mapper/components/mapInterface/widgets';
|
||||||
|
|
||||||
|
export const CURRENT_WINDOWS_VERSION = 8;
|
||||||
|
export const WINDOWS_LOCAL_STORE_KEY = 'windows:settings:v2';
|
||||||
|
|
||||||
|
export enum WidgetsIds {
|
||||||
|
info = 'info',
|
||||||
|
signatures = 'signatures',
|
||||||
|
local = 'local',
|
||||||
|
routes = 'routes',
|
||||||
|
structures = 'structures',
|
||||||
|
kills = 'kills',
|
||||||
|
}
|
||||||
|
|
||||||
|
export const STORED_VISIBLE_WIDGETS_DEFAULT = [
|
||||||
|
WidgetsIds.info,
|
||||||
|
WidgetsIds.local,
|
||||||
|
WidgetsIds.routes,
|
||||||
|
WidgetsIds.signatures,
|
||||||
|
];
|
||||||
|
|
||||||
|
export const DEFAULT_WIDGETS: WindowProps[] = [
|
||||||
|
{
|
||||||
|
id: WidgetsIds.info,
|
||||||
|
position: { x: 10, y: 10 },
|
||||||
|
size: { width: 250, height: 200 },
|
||||||
|
zIndex: 0,
|
||||||
|
content: () => <SystemInfo />,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.signatures,
|
||||||
|
position: { x: 10, y: 220 },
|
||||||
|
size: { width: 250, height: 300 },
|
||||||
|
zIndex: 0,
|
||||||
|
content: () => <SystemSignatures />,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.local,
|
||||||
|
position: { x: 270, y: 10 },
|
||||||
|
size: { width: 250, height: 510 },
|
||||||
|
zIndex: 0,
|
||||||
|
content: () => <LocalCharacters />,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.routes,
|
||||||
|
position: { x: 10, y: 530 },
|
||||||
|
size: { width: 510, height: 200 },
|
||||||
|
zIndex: 0,
|
||||||
|
content: () => <RoutesWidget />,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.structures,
|
||||||
|
position: { x: 10, y: 730 },
|
||||||
|
size: { width: 510, height: 200 },
|
||||||
|
zIndex: 0,
|
||||||
|
content: () => <SystemStructures />,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.kills,
|
||||||
|
position: { x: 270, y: 730 },
|
||||||
|
size: { width: 510, height: 200 },
|
||||||
|
zIndex: 0,
|
||||||
|
content: () => <SystemKills />,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
type WidgetsCheckboxesType = {
|
||||||
|
id: WidgetsIds;
|
||||||
|
label: string;
|
||||||
|
}[];
|
||||||
|
|
||||||
|
export const WIDGETS_CHECKBOXES_PROPS: WidgetsCheckboxesType = [
|
||||||
|
{
|
||||||
|
id: WidgetsIds.info,
|
||||||
|
label: 'System Info',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.signatures,
|
||||||
|
label: 'Signatures',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.local,
|
||||||
|
label: 'Local',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.routes,
|
||||||
|
label: 'Routes',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.structures,
|
||||||
|
label: 'Structures',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: WidgetsIds.kills,
|
||||||
|
label: 'Kills',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export function getWidgetsCheckboxesProps(detailedKillsDisabled: boolean): WidgetsCheckboxesType {
|
||||||
|
return filterOutKills(WIDGETS_CHECKBOXES_PROPS, detailedKillsDisabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function filterOutKills<T extends { id: WidgetsIds }>(items: T[], shouldFilter: boolean) {
|
||||||
|
if (!shouldFilter) return items;
|
||||||
|
return items.filter((w) => w.id !== WidgetsIds.kills);
|
||||||
|
}
|
||||||
@@ -8,7 +8,6 @@
|
|||||||
.RouteSystem {
|
.RouteSystem {
|
||||||
width: 8px;
|
width: 8px;
|
||||||
height: 8px;
|
height: 8px;
|
||||||
background: #ffffff;
|
|
||||||
|
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: opacity 200ms;
|
transition: opacity 200ms;
|
||||||
|
|||||||
@@ -0,0 +1,85 @@
|
|||||||
|
import React, { useMemo, useState } from 'react';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { Widget } from '@/hooks/Mapper/components/mapInterface/components';
|
||||||
|
import { SystemKillsContent } from './SystemKillsContent/SystemKillsContent';
|
||||||
|
import { KillsHeader } from './components/SystemKillsHeader';
|
||||||
|
import { useKillsWidgetSettings } from './hooks/useKillsWidgetSettings';
|
||||||
|
import { useSystemKills } from './hooks/useSystemKills';
|
||||||
|
import { KillsSettingsDialog } from './components/SystemKillsSettingsDialog';
|
||||||
|
|
||||||
|
export const SystemKills: React.FC = () => {
|
||||||
|
const {
|
||||||
|
data: { selectedSystems, systems },
|
||||||
|
outCommand,
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const [systemId] = selectedSystems || [];
|
||||||
|
|
||||||
|
const [settingsDialogVisible, setSettingsDialogVisible] = useState(false);
|
||||||
|
|
||||||
|
const systemNameMap = useMemo(() => {
|
||||||
|
const map: Record<string, string> = {};
|
||||||
|
systems.forEach(sys => {
|
||||||
|
map[sys.id] = sys.temporary_name || sys.name || '???';
|
||||||
|
});
|
||||||
|
return map;
|
||||||
|
}, [systems]);
|
||||||
|
|
||||||
|
const [settings] = useKillsWidgetSettings();
|
||||||
|
const visible = settings.showAll;
|
||||||
|
|
||||||
|
const { kills, isLoading, error } = useSystemKills({
|
||||||
|
systemId,
|
||||||
|
outCommand,
|
||||||
|
showAllVisible: visible,
|
||||||
|
});
|
||||||
|
|
||||||
|
const isNothingSelected = !systemId && !visible;
|
||||||
|
const showLoading = isLoading && kills.length === 0;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="h-full flex flex-col min-h-0">
|
||||||
|
<div className="flex flex-col flex-1 min-h-0">
|
||||||
|
<Widget label={<KillsHeader systemId={systemId} onOpenSettings={() => setSettingsDialogVisible(true)} />}>
|
||||||
|
{isNothingSelected && (
|
||||||
|
<div className="w-full h-full flex justify-center items-center select-none text-center text-stone-400/80 text-sm">
|
||||||
|
No system selected (or toggle “Show all systems”)
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!isNothingSelected && showLoading && (
|
||||||
|
<div className="w-full h-full flex justify-center items-center select-none text-center text-stone-400/80 text-sm">
|
||||||
|
Loading Kills...
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!isNothingSelected && !showLoading && error && (
|
||||||
|
<div className="w-full h-full flex justify-center items-center select-none text-center text-red-400 text-sm">
|
||||||
|
{error}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!isNothingSelected && !showLoading && !error && (!kills || kills.length === 0) && (
|
||||||
|
<div className="w-full h-full flex justify-center items-center select-none text-center text-stone-400/80 text-sm">
|
||||||
|
No kills found
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!isNothingSelected && !showLoading && !error && (
|
||||||
|
<div className="flex-1 flex flex-col overflow-y-auto">
|
||||||
|
<SystemKillsContent
|
||||||
|
key={settings.compact ? 'compact' : 'normal'}
|
||||||
|
kills={kills}
|
||||||
|
systemNameMap={systemNameMap}
|
||||||
|
compact={settings.compact}
|
||||||
|
onlyOneSystem={!visible}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Widget>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<KillsSettingsDialog visible={settingsDialogVisible} setVisible={setSettingsDialogVisible} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,16 @@
|
|||||||
|
.TableRowCompact {
|
||||||
|
height: 8px;
|
||||||
|
max-height: 8px;
|
||||||
|
font-size: 12px !important;
|
||||||
|
line-height: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.Table {
|
||||||
|
font-size: 12px;
|
||||||
|
border-collapse: collapse;
|
||||||
|
}
|
||||||
|
|
||||||
|
.Tooltip {
|
||||||
|
white-space: pre-line;
|
||||||
|
line-height: 1.2rem;
|
||||||
|
}
|
||||||
@@ -0,0 +1,50 @@
|
|||||||
|
import React, { useMemo } from 'react';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { DetailedKill } from '@/hooks/Mapper/types/kills';
|
||||||
|
import { KillRow } from '../components/SystemKillsRow';
|
||||||
|
|
||||||
|
interface SystemKillsContentProps {
|
||||||
|
kills: DetailedKill[];
|
||||||
|
systemNameMap: Record<string, string>;
|
||||||
|
compact?: boolean;
|
||||||
|
onlyOneSystem?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SystemKillsContent: React.FC<SystemKillsContentProps> = ({
|
||||||
|
kills,
|
||||||
|
systemNameMap,
|
||||||
|
compact = false,
|
||||||
|
onlyOneSystem = false,
|
||||||
|
}) => {
|
||||||
|
const sortedKills = useMemo(() => {
|
||||||
|
return [...kills].sort((a, b) => {
|
||||||
|
const timeA = a.kill_time ? new Date(a.kill_time).getTime() : 0;
|
||||||
|
const timeB = b.kill_time ? new Date(b.kill_time).getTime() : 0;
|
||||||
|
return timeB - timeA;
|
||||||
|
});
|
||||||
|
}, [kills]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
'flex flex-col w-full text-stone-200 text-xs transition-all duration-300',
|
||||||
|
compact ? 'p-1' : 'p-1',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{sortedKills.map(kill => {
|
||||||
|
const systemIdStr = String(kill.solar_system_id);
|
||||||
|
const systemName = systemNameMap[systemIdStr] || `System ${systemIdStr}`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<KillRow
|
||||||
|
key={kill.killmail_id}
|
||||||
|
killDetails={kill}
|
||||||
|
systemName={systemName}
|
||||||
|
isCompact={compact}
|
||||||
|
onlyOneSystem={onlyOneSystem}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { zkillLink } from '../helpers';
|
||||||
|
import classes from './SystemKillRow.module.scss';
|
||||||
|
|
||||||
|
interface AttackerRowSubInfoProps {
|
||||||
|
finalBlowCharId: number | null | undefined;
|
||||||
|
finalBlowCharName?: string;
|
||||||
|
attackerPortraitUrl: string | null;
|
||||||
|
|
||||||
|
finalBlowCorpId: number | null | undefined;
|
||||||
|
finalBlowCorpName?: string;
|
||||||
|
attackerCorpLogoUrl: string | null;
|
||||||
|
|
||||||
|
finalBlowAllianceId: number | null | undefined;
|
||||||
|
finalBlowAllianceName?: string;
|
||||||
|
attackerAllianceLogoUrl: string | null;
|
||||||
|
|
||||||
|
containerHeight?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const AttackerRowSubInfo: React.FC<AttackerRowSubInfoProps> = ({
|
||||||
|
finalBlowCharId = 0,
|
||||||
|
finalBlowCharName,
|
||||||
|
attackerPortraitUrl,
|
||||||
|
containerHeight = 8,
|
||||||
|
}) => {
|
||||||
|
if (!attackerPortraitUrl || finalBlowCharId === null || finalBlowCharId <= 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const containerClass = `h-${containerHeight}`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={clsx('flex items-start gap-1', containerClass)}>
|
||||||
|
<div className="relative shrink-0 w-auto h-full overflow-hidden">
|
||||||
|
<a
|
||||||
|
href={zkillLink('character', finalBlowCharId)}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="block h-full"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
src={attackerPortraitUrl}
|
||||||
|
alt={finalBlowCharName || 'AttackerPortrait'}
|
||||||
|
className={clsx(classes.killRowImage, 'h-full w-auto object-contain')}
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,180 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { DetailedKill } from '@/hooks/Mapper/types/kills';
|
||||||
|
import {
|
||||||
|
formatISK,
|
||||||
|
formatTimeMixed,
|
||||||
|
zkillLink,
|
||||||
|
getAttackerSubscript,
|
||||||
|
buildVictimImageUrls,
|
||||||
|
buildAttackerImageUrls,
|
||||||
|
getPrimaryLogoAndTooltip,
|
||||||
|
getAttackerPrimaryImageAndTooltip,
|
||||||
|
} from '../helpers';
|
||||||
|
import { WdTooltipWrapper } from '../../../../ui-kit/WdTooltipWrapper';
|
||||||
|
import classes from './SystemKillRow.module.scss';
|
||||||
|
import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
|
export interface CompactKillRowProps {
|
||||||
|
killDetails: DetailedKill;
|
||||||
|
systemName: string;
|
||||||
|
onlyOneSystem: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const CompactKillRow: React.FC<CompactKillRowProps> = ({ killDetails, systemName, onlyOneSystem }) => {
|
||||||
|
const {
|
||||||
|
killmail_id = 0,
|
||||||
|
|
||||||
|
victim_char_name = 'Unknown Pilot',
|
||||||
|
victim_alliance_ticker = '',
|
||||||
|
victim_corp_ticker = '',
|
||||||
|
victim_ship_name = 'Unknown Ship',
|
||||||
|
victim_corp_name = '',
|
||||||
|
victim_alliance_name = '',
|
||||||
|
victim_char_id = 0,
|
||||||
|
victim_corp_id = 0,
|
||||||
|
victim_alliance_id = 0,
|
||||||
|
victim_ship_type_id = 0,
|
||||||
|
|
||||||
|
final_blow_char_id = 0,
|
||||||
|
final_blow_char_name = '',
|
||||||
|
final_blow_alliance_ticker = '',
|
||||||
|
final_blow_alliance_name = '',
|
||||||
|
final_blow_alliance_id = 0,
|
||||||
|
final_blow_corp_ticker = '',
|
||||||
|
final_blow_corp_id = 0,
|
||||||
|
final_blow_corp_name = '',
|
||||||
|
final_blow_ship_type_id = 0,
|
||||||
|
|
||||||
|
kill_time = '',
|
||||||
|
total_value = 0,
|
||||||
|
} = killDetails || {};
|
||||||
|
|
||||||
|
const attackerIsNpc = final_blow_char_id === 0;
|
||||||
|
const victimAffiliationTicker = victim_alliance_ticker || victim_corp_ticker || 'No Ticker';
|
||||||
|
const killValueFormatted = total_value != null && total_value > 0 ? `${formatISK(total_value)} ISK` : null;
|
||||||
|
const attackerName = attackerIsNpc ? '' : final_blow_char_name;
|
||||||
|
const attackerTicker = attackerIsNpc ? '' : final_blow_alliance_ticker || final_blow_corp_ticker || '';
|
||||||
|
const killTimeAgo = kill_time ? formatTimeMixed(kill_time) : '0h ago';
|
||||||
|
const attackerSubscript = getAttackerSubscript(killDetails);
|
||||||
|
|
||||||
|
const { victimCorpLogoUrl, victimAllianceLogoUrl } = buildVictimImageUrls({
|
||||||
|
victim_char_id,
|
||||||
|
victim_ship_type_id,
|
||||||
|
victim_corp_id,
|
||||||
|
victim_alliance_id,
|
||||||
|
});
|
||||||
|
const { attackerCorpLogoUrl, attackerAllianceLogoUrl } = buildAttackerImageUrls({
|
||||||
|
final_blow_char_id,
|
||||||
|
final_blow_corp_id,
|
||||||
|
final_blow_alliance_id,
|
||||||
|
});
|
||||||
|
|
||||||
|
const { url: victimPrimaryLogoUrl, tooltip: victimPrimaryTooltip } = getPrimaryLogoAndTooltip(
|
||||||
|
victimAllianceLogoUrl,
|
||||||
|
victimCorpLogoUrl,
|
||||||
|
victim_alliance_name,
|
||||||
|
victim_corp_name,
|
||||||
|
'Victim',
|
||||||
|
);
|
||||||
|
|
||||||
|
const { url: attackerPrimaryImageUrl, tooltip: attackerPrimaryTooltip } = getAttackerPrimaryImageAndTooltip(
|
||||||
|
attackerIsNpc,
|
||||||
|
attackerAllianceLogoUrl,
|
||||||
|
attackerCorpLogoUrl,
|
||||||
|
final_blow_alliance_name,
|
||||||
|
final_blow_corp_name,
|
||||||
|
final_blow_ship_type_id || 0,
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
'h-10 flex items-center border-b border-stone-800',
|
||||||
|
'text-xs whitespace-nowrap overflow-hidden leading-none',
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{victimPrimaryLogoUrl && (
|
||||||
|
<WdTooltipWrapper content={victimPrimaryTooltip} position={TooltipPosition.top}>
|
||||||
|
<a
|
||||||
|
href={zkillLink('kill', killmail_id)}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="relative shrink-0 w-8 h-8 overflow-hidden"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
src={victimPrimaryLogoUrl}
|
||||||
|
alt="VictimPrimaryLogo"
|
||||||
|
className={clsx(classes.killRowImage, 'w-full h-full object-contain')}
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<div className="flex flex-col ml-2 min-w-0 overflow-hidden leading-[1rem]">
|
||||||
|
<div className="truncate text-stone-200">
|
||||||
|
{victim_char_name}
|
||||||
|
<span className="text-stone-400"> / {victimAffiliationTicker}</span>
|
||||||
|
</div>
|
||||||
|
<div className="truncate text-stone-300">
|
||||||
|
{victim_ship_name}
|
||||||
|
{killValueFormatted && (
|
||||||
|
<>
|
||||||
|
<span className="ml-1 text-stone-400">/</span>
|
||||||
|
<span className="ml-1 text-green-400">{killValueFormatted}</span>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex items-center ml-auto gap-2">
|
||||||
|
<div className="flex flex-col items-end min-w-0 overflow-hidden text-right leading-[1rem]">
|
||||||
|
{!attackerIsNpc && (attackerName || attackerTicker) && (
|
||||||
|
<div className="truncate text-stone-200">
|
||||||
|
{attackerName}
|
||||||
|
{attackerTicker && <span className="ml-1 text-stone-400">/ {attackerTicker}</span>}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<div className="truncate text-stone-400">
|
||||||
|
{!onlyOneSystem && systemName ? (
|
||||||
|
<>
|
||||||
|
{systemName} / <span className="ml-1 text-red-400">{killTimeAgo}</span>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<span className="text-red-400">{killTimeAgo}</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{attackerPrimaryImageUrl && (
|
||||||
|
<WdTooltipWrapper content={attackerPrimaryTooltip} position={TooltipPosition.top}>
|
||||||
|
<a
|
||||||
|
href={zkillLink('kill', killmail_id)}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="relative shrink-0 w-8 h-8 overflow-hidden"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
src={attackerPrimaryImageUrl}
|
||||||
|
alt={attackerIsNpc ? 'NpcShip' : 'AttackerPrimaryLogo'}
|
||||||
|
className={clsx(classes.killRowImage, 'w-full h-full object-contain')}
|
||||||
|
/>
|
||||||
|
{attackerSubscript && (
|
||||||
|
<span
|
||||||
|
className={clsx(
|
||||||
|
classes.attackerCountLabel,
|
||||||
|
attackerSubscript.cssClass,
|
||||||
|
'text-[0.6rem] leading-none px-[2px]',
|
||||||
|
)}
|
||||||
|
style={{ bottom: 0, right: 0 }}
|
||||||
|
>
|
||||||
|
{attackerSubscript.label}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</a>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,197 @@
|
|||||||
|
// FullKillRow.tsx
|
||||||
|
import React from 'react';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { DetailedKill } from '@/hooks/Mapper/types/kills';
|
||||||
|
import {
|
||||||
|
formatISK,
|
||||||
|
formatTimeMixed,
|
||||||
|
zkillLink,
|
||||||
|
getAttackerSubscript,
|
||||||
|
buildVictimImageUrls,
|
||||||
|
buildAttackerImageUrls,
|
||||||
|
getPrimaryLogoAndTooltip,
|
||||||
|
getAttackerPrimaryImageAndTooltip,
|
||||||
|
} from '../helpers';
|
||||||
|
import { VictimRowSubInfo } from './VictimRowSubInfo';
|
||||||
|
import { WdTooltipWrapper } from '../../../../ui-kit/WdTooltipWrapper';
|
||||||
|
import classes from './SystemKillRow.module.scss';
|
||||||
|
import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
|
export interface FullKillRowProps {
|
||||||
|
killDetails: DetailedKill;
|
||||||
|
systemName: string;
|
||||||
|
onlyOneSystem: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const FullKillRow: React.FC<FullKillRowProps> = ({ killDetails, systemName, onlyOneSystem }) => {
|
||||||
|
const {
|
||||||
|
killmail_id = 0,
|
||||||
|
|
||||||
|
victim_char_name = '',
|
||||||
|
victim_alliance_ticker = '',
|
||||||
|
victim_corp_ticker = '',
|
||||||
|
victim_ship_name = '',
|
||||||
|
victim_char_id = 0,
|
||||||
|
victim_corp_id = 0,
|
||||||
|
victim_alliance_id = 0,
|
||||||
|
victim_ship_type_id = 0,
|
||||||
|
victim_corp_name = '',
|
||||||
|
victim_alliance_name = '',
|
||||||
|
|
||||||
|
final_blow_char_id = 0,
|
||||||
|
final_blow_char_name = '',
|
||||||
|
final_blow_alliance_ticker = '',
|
||||||
|
final_blow_corp_ticker = '',
|
||||||
|
final_blow_corp_name = '',
|
||||||
|
final_blow_alliance_name = '',
|
||||||
|
final_blow_corp_id = 0,
|
||||||
|
final_blow_alliance_id = 0,
|
||||||
|
final_blow_ship_name = '',
|
||||||
|
final_blow_ship_type_id = 0,
|
||||||
|
|
||||||
|
total_value = 0,
|
||||||
|
kill_time = '',
|
||||||
|
} = killDetails || {};
|
||||||
|
|
||||||
|
const attackerIsNpc = final_blow_char_id === 0;
|
||||||
|
const victimAffiliation = victim_alliance_ticker || victim_corp_ticker;
|
||||||
|
const attackerAffiliation = attackerIsNpc ? '' : final_blow_alliance_ticker || final_blow_corp_ticker || '';
|
||||||
|
|
||||||
|
const killValueFormatted = total_value !== null && total_value > 0 ? `${formatISK(total_value)} ISK` : null;
|
||||||
|
const killTimeAgo = kill_time ? formatTimeMixed(kill_time) : '0h ago';
|
||||||
|
|
||||||
|
const { victimPortraitUrl, victimCorpLogoUrl, victimAllianceLogoUrl } = buildVictimImageUrls({
|
||||||
|
victim_char_id,
|
||||||
|
victim_ship_type_id,
|
||||||
|
victim_corp_id,
|
||||||
|
victim_alliance_id,
|
||||||
|
});
|
||||||
|
const { attackerPortraitUrl, attackerCorpLogoUrl, attackerAllianceLogoUrl } = buildAttackerImageUrls({
|
||||||
|
final_blow_char_id,
|
||||||
|
final_blow_corp_id,
|
||||||
|
final_blow_alliance_id,
|
||||||
|
});
|
||||||
|
|
||||||
|
const { url: victimPrimaryImageUrl, tooltip: victimPrimaryTooltip } = getPrimaryLogoAndTooltip(
|
||||||
|
victimAllianceLogoUrl,
|
||||||
|
victimCorpLogoUrl,
|
||||||
|
victim_alliance_name,
|
||||||
|
victim_corp_name,
|
||||||
|
'Victim',
|
||||||
|
);
|
||||||
|
|
||||||
|
const { url: attackerPrimaryImageUrl, tooltip: attackerPrimaryTooltip } = getAttackerPrimaryImageAndTooltip(
|
||||||
|
attackerIsNpc,
|
||||||
|
attackerAllianceLogoUrl,
|
||||||
|
attackerCorpLogoUrl,
|
||||||
|
final_blow_alliance_name,
|
||||||
|
final_blow_corp_name,
|
||||||
|
final_blow_ship_type_id || 0,
|
||||||
|
);
|
||||||
|
|
||||||
|
const attackerSubscript = getAttackerSubscript(killDetails);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={clsx(classes.killRowContainer, 'h-18 w-full justify-between items-start text-sm py-[4px]')}>
|
||||||
|
{/* ---------------- Victim Side ---------------- */}
|
||||||
|
<div className="flex items-start gap-1 min-w-0 h-full">
|
||||||
|
{/* Victim top-level logo (corp or alliance), with tooltip */}
|
||||||
|
{victimPrimaryImageUrl && (
|
||||||
|
<WdTooltipWrapper content={victimPrimaryTooltip} position={TooltipPosition.top}>
|
||||||
|
<div className="relative shrink-0 w-14 h-14 overflow-hidden">
|
||||||
|
<a
|
||||||
|
href={zkillLink('kill', killmail_id)}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="block w-full h-full"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
src={victimPrimaryImageUrl}
|
||||||
|
alt="VictimPrimaryLogo"
|
||||||
|
className={clsx(classes.killRowImage, 'w-full h-full object-contain')}
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
)}
|
||||||
|
<VictimRowSubInfo
|
||||||
|
victimCharName={victim_char_name}
|
||||||
|
victimCharacterId={victim_char_id}
|
||||||
|
victimPortraitUrl={victimPortraitUrl}
|
||||||
|
/>
|
||||||
|
<div className="flex flex-col text-stone-200 leading-4 min-w-0 overflow-hidden">
|
||||||
|
<div className="truncate">
|
||||||
|
<span className="font-semibold">{victim_char_name}</span>
|
||||||
|
{victimAffiliation && <span className="ml-1 text-stone-400">/ {victimAffiliation}</span>}
|
||||||
|
</div>
|
||||||
|
<div className="truncate text-stone-300">
|
||||||
|
{victim_ship_name}
|
||||||
|
{killValueFormatted && (
|
||||||
|
<>
|
||||||
|
<span className="ml-1 text-stone-400">/</span>
|
||||||
|
<span className="ml-1 text-green-400">{killValueFormatted}</span>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="truncate text-stone-400">{!onlyOneSystem && systemName && <span>{systemName}</span>}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex items-start gap-1 min-w-0 h-full">
|
||||||
|
<div className="flex flex-col items-end leading-4 min-w-0 overflow-hidden text-right">
|
||||||
|
{!attackerIsNpc && (
|
||||||
|
<div className="truncate font-semibold">
|
||||||
|
{final_blow_char_name}
|
||||||
|
{attackerAffiliation && <span className="ml-1 text-stone-400">/ {attackerAffiliation}</span>}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{!attackerIsNpc && final_blow_ship_name && (
|
||||||
|
<div className="truncate text-stone-300">{final_blow_ship_name}</div>
|
||||||
|
)}
|
||||||
|
<div className="truncate text-red-400">{killTimeAgo}</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{!attackerIsNpc && attackerPortraitUrl && final_blow_char_id !== null && final_blow_char_id > 0 && (
|
||||||
|
<div className="relative shrink-0 w-14 h-14 overflow-hidden">
|
||||||
|
<a
|
||||||
|
href={zkillLink('character', final_blow_char_id)}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="block w-full h-full"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
src={attackerPortraitUrl}
|
||||||
|
alt="AttackerPortrait"
|
||||||
|
className={clsx(classes.killRowImage, 'w-full h-full object-contain')}
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{attackerPrimaryImageUrl && (
|
||||||
|
<WdTooltipWrapper content={attackerPrimaryTooltip} position={TooltipPosition.top}>
|
||||||
|
<div className="relative shrink-0 w-14 h-14 overflow-hidden">
|
||||||
|
<a
|
||||||
|
href={zkillLink('kill', killmail_id)}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="block w-full h-full"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
src={attackerPrimaryImageUrl}
|
||||||
|
alt={attackerIsNpc ? 'NpcShip' : 'AttackerPrimaryLogo'}
|
||||||
|
className={clsx(classes.killRowImage, 'w-full h-full object-contain')}
|
||||||
|
/>
|
||||||
|
{attackerSubscript && (
|
||||||
|
<span className={clsx(attackerSubscript.cssClass, classes.attackerCountLabel)}>
|
||||||
|
{attackerSubscript.label}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
.killRowContainer {
|
||||||
|
@apply flex items-center whitespace-nowrap overflow-hidden;
|
||||||
|
&:not(:last-child) {
|
||||||
|
@apply border-b border-stone-800;
|
||||||
|
}
|
||||||
|
@apply bg-transparent transition-all hover:bg-stone-900 hover:border-stone-700;
|
||||||
|
}
|
||||||
|
|
||||||
|
.killRowImage {
|
||||||
|
@apply border border-stone-800 rounded-[4px] object-contain;
|
||||||
|
}
|
||||||
|
|
||||||
|
.attackerCountLabel {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
right: 0;
|
||||||
|
font-size: 10px;
|
||||||
|
padding: 0 2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.attackerCountLabelCompact {
|
||||||
|
position: absolute;
|
||||||
|
left: 0;
|
||||||
|
bottom: 0;
|
||||||
|
font-size: 0.6rem;
|
||||||
|
line-height: 1;
|
||||||
|
background-color: rgba(0, 0, 0, 0.7);
|
||||||
|
padding: 1px 2px;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import {
|
||||||
|
LayoutEventBlocker,
|
||||||
|
WdCheckbox,
|
||||||
|
WdImgButton,
|
||||||
|
TooltipPosition,
|
||||||
|
SystemView,
|
||||||
|
} from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { useKillsWidgetSettings } from '../hooks/useKillsWidgetSettings';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
|
||||||
|
interface KillsWidgetHeaderProps {
|
||||||
|
systemId?: string;
|
||||||
|
onOpenSettings: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const KillsHeader: React.FC<KillsWidgetHeaderProps> = ({ systemId, onOpenSettings }) => {
|
||||||
|
const [settings, setSettings] = useKillsWidgetSettings();
|
||||||
|
const { showAll } = settings;
|
||||||
|
|
||||||
|
const onToggleShowAllVisible = () => {
|
||||||
|
setSettings(prev => ({ ...prev, showAll: !prev.showAll }));
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex justify-between items-center text-xs w-full">
|
||||||
|
<div className="flex items-center gap-1">
|
||||||
|
<div className="text-stone-400">
|
||||||
|
Kills
|
||||||
|
{systemId && !showAll && ' in '}
|
||||||
|
</div>
|
||||||
|
{systemId && !showAll && <SystemView systemId={systemId} className="select-none text-center" hideRegion />}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<LayoutEventBlocker className="flex gap-2 items-center">
|
||||||
|
<WdCheckbox
|
||||||
|
size="xs"
|
||||||
|
labelSide="left"
|
||||||
|
label="Show all systems"
|
||||||
|
value={showAll}
|
||||||
|
classNameLabel="text-stone-400 hover:text-stone-200 transition duration-300"
|
||||||
|
onChange={onToggleShowAllVisible}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<WdImgButton
|
||||||
|
className={PrimeIcons.SLIDERS_H}
|
||||||
|
onClick={onOpenSettings}
|
||||||
|
tooltip={{
|
||||||
|
content: 'Open Kills Settings',
|
||||||
|
position: TooltipPosition.left,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</LayoutEventBlocker>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { DetailedKill } from '@/hooks/Mapper/types/kills';
|
||||||
|
import { CompactKillRow } from './CompactKillRow';
|
||||||
|
import { FullKillRow } from './FullKillRow';
|
||||||
|
|
||||||
|
export interface KillRowProps {
|
||||||
|
killDetails: DetailedKill;
|
||||||
|
systemName: string;
|
||||||
|
isCompact?: boolean;
|
||||||
|
onlyOneSystem?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const KillRow: React.FC<KillRowProps> = ({
|
||||||
|
killDetails,
|
||||||
|
systemName,
|
||||||
|
isCompact = false,
|
||||||
|
onlyOneSystem = false,
|
||||||
|
}) => {
|
||||||
|
if (isCompact) {
|
||||||
|
return <CompactKillRow killDetails={killDetails} systemName={systemName} onlyOneSystem={onlyOneSystem} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return <FullKillRow killDetails={killDetails} systemName={systemName} onlyOneSystem={onlyOneSystem} />;
|
||||||
|
};
|
||||||
@@ -0,0 +1,136 @@
|
|||||||
|
import React, { useCallback, useEffect, useRef, useState } from 'react';
|
||||||
|
import { Dialog } from 'primereact/dialog';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { WdImgButton, SystemView, TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { useKillsWidgetSettings } from '../hooks/useKillsWidgetSettings';
|
||||||
|
import {
|
||||||
|
AddSystemDialog,
|
||||||
|
SearchOnSubmitCallback,
|
||||||
|
} from '@/hooks/Mapper/components/mapInterface/components/AddSystemDialog';
|
||||||
|
|
||||||
|
interface KillsSettingsDialogProps {
|
||||||
|
visible: boolean;
|
||||||
|
setVisible: (visible: boolean) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const KillsSettingsDialog: React.FC<KillsSettingsDialogProps> = ({ visible, setVisible }) => {
|
||||||
|
const [globalSettings, setGlobalSettings] = useKillsWidgetSettings();
|
||||||
|
const localRef = useRef({
|
||||||
|
compact: globalSettings.compact,
|
||||||
|
showAll: globalSettings.showAll,
|
||||||
|
excludedSystems: globalSettings.excludedSystems || [],
|
||||||
|
});
|
||||||
|
|
||||||
|
const [, forceRender] = useState(0);
|
||||||
|
|
||||||
|
const [addSystemDialogVisible, setAddSystemDialogVisible] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (visible) {
|
||||||
|
localRef.current = {
|
||||||
|
compact: globalSettings.compact,
|
||||||
|
showAll: globalSettings.showAll,
|
||||||
|
excludedSystems: globalSettings.excludedSystems || [],
|
||||||
|
};
|
||||||
|
forceRender(n => n + 1);
|
||||||
|
}
|
||||||
|
}, [visible, globalSettings]);
|
||||||
|
|
||||||
|
const handleCompactChange = useCallback((checked: boolean) => {
|
||||||
|
localRef.current = {
|
||||||
|
...localRef.current,
|
||||||
|
compact: checked,
|
||||||
|
};
|
||||||
|
forceRender(n => n + 1);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleRemoveSystem = useCallback((sysId: number) => {
|
||||||
|
localRef.current = {
|
||||||
|
...localRef.current,
|
||||||
|
excludedSystems: localRef.current.excludedSystems.filter(id => id !== sysId),
|
||||||
|
};
|
||||||
|
forceRender(n => n + 1);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleAddSystemSubmit: SearchOnSubmitCallback = useCallback(item => {
|
||||||
|
if (localRef.current.excludedSystems.includes(item.value)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
localRef.current = {
|
||||||
|
...localRef.current,
|
||||||
|
excludedSystems: [...localRef.current.excludedSystems, item.value],
|
||||||
|
};
|
||||||
|
forceRender(n => n + 1);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleApply = useCallback(() => {
|
||||||
|
setGlobalSettings(prev => ({
|
||||||
|
...prev,
|
||||||
|
...localRef.current,
|
||||||
|
}));
|
||||||
|
setVisible(false);
|
||||||
|
}, [setGlobalSettings, setVisible]);
|
||||||
|
|
||||||
|
const handleHide = useCallback(() => {
|
||||||
|
setVisible(false);
|
||||||
|
}, [setVisible]);
|
||||||
|
|
||||||
|
const localData = localRef.current;
|
||||||
|
const excluded = localData.excludedSystems || [];
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog header="Kills Settings" visible={visible} style={{ width: '440px' }} draggable={false} onHide={handleHide}>
|
||||||
|
<div className="flex flex-col gap-3 p-2.5">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
id="kills-compact-mode"
|
||||||
|
checked={localData.compact}
|
||||||
|
onChange={e => handleCompactChange(e.target.checked)}
|
||||||
|
/>
|
||||||
|
<label htmlFor="kills-compact-mode" className="cursor-pointer">
|
||||||
|
Use compact mode
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<label className="text-sm text-stone-400">Excluded Systems</label>
|
||||||
|
<WdImgButton
|
||||||
|
className={PrimeIcons.PLUS_CIRCLE}
|
||||||
|
onClick={() => setAddSystemDialogVisible(true)}
|
||||||
|
tooltip={{ content: 'Add system to excluded list' }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{excluded.length === 0 && <div className="text-stone-500 text-xs italic">No systems excluded.</div>}
|
||||||
|
{excluded.map(sysId => (
|
||||||
|
<div key={sysId} className="flex items-center justify-between border-b border-stone-600 py-1 px-1 text-xs">
|
||||||
|
<SystemView systemId={sysId.toString()} hideRegion compact />
|
||||||
|
|
||||||
|
<WdImgButton
|
||||||
|
className={PrimeIcons.TRASH}
|
||||||
|
onClick={() => handleRemoveSystem(sysId)}
|
||||||
|
tooltip={{ content: 'Remove from excluded', position: TooltipPosition.top }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Apply + Close button row */}
|
||||||
|
<div className="flex gap-2 justify-end mt-4">
|
||||||
|
<Button onClick={handleApply} label="Apply" outlined size="small" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* AddSystemDialog for picking new systems to exclude */}
|
||||||
|
<AddSystemDialog
|
||||||
|
title="Add system to kills exclude list"
|
||||||
|
visible={addSystemDialogVisible}
|
||||||
|
setVisible={() => setAddSystemDialogVisible(false)}
|
||||||
|
onSubmit={handleAddSystemSubmit}
|
||||||
|
excludedSystems={excluded}
|
||||||
|
/>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
// VictimSubRowInfo.tsx
|
||||||
|
import React from 'react';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { zkillLink } from '../helpers';
|
||||||
|
import classes from './SystemKillRow.module.scss';
|
||||||
|
|
||||||
|
interface VictimRowSubInfoProps {
|
||||||
|
victimCharacterId: number | null;
|
||||||
|
victimPortraitUrl: string | null;
|
||||||
|
victimCharName?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const VictimRowSubInfo: React.FC<VictimRowSubInfoProps> = ({
|
||||||
|
victimCharacterId = 0,
|
||||||
|
victimPortraitUrl,
|
||||||
|
victimCharName,
|
||||||
|
}) => {
|
||||||
|
if (!victimPortraitUrl || victimCharacterId === null || victimCharacterId <= 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex items-start gap-1 h-14">
|
||||||
|
<div className="relative shrink-0 w-14 h-14 overflow-hidden">
|
||||||
|
<a
|
||||||
|
href={zkillLink('character', victimCharacterId)}
|
||||||
|
target="_blank"
|
||||||
|
rel="noopener noreferrer"
|
||||||
|
className="block w-full h-full"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
src={victimPortraitUrl}
|
||||||
|
alt={victimCharName || 'Victim Portrait'}
|
||||||
|
className={clsx(classes.killRowImage, 'w-full h-full object-contain')}
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,2 @@
|
|||||||
|
export * from './linkHelpers';
|
||||||
|
export * from './killRowUtils';
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
import { DetailedKill } from '@/hooks/Mapper/types/kills';
|
||||||
|
|
||||||
|
/** Returns "5m ago", "3h ago", "2.5d ago", etc. */
|
||||||
|
export function formatTimeMixed(killTime: string): string {
|
||||||
|
const killDate = new Date(killTime);
|
||||||
|
const diffMs = Date.now() - killDate.getTime();
|
||||||
|
const diffHours = diffMs / (1000 * 60 * 60);
|
||||||
|
|
||||||
|
if (diffHours < 1) {
|
||||||
|
const mins = Math.round(diffHours * 60);
|
||||||
|
return `${mins}m ago`;
|
||||||
|
} else if (diffHours < 24) {
|
||||||
|
const hours = Math.round(diffHours);
|
||||||
|
return `${hours}h ago`;
|
||||||
|
} else {
|
||||||
|
const days = diffHours / 24;
|
||||||
|
const roundedDays = days.toFixed(1);
|
||||||
|
return `${roundedDays}d ago`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Formats integer ISK values into k/M/B/T. */
|
||||||
|
export function formatISK(value: number): string {
|
||||||
|
if (value >= 1_000_000_000_000) {
|
||||||
|
return `${(value / 1_000_000_000_000).toFixed(2)}T`;
|
||||||
|
} else if (value >= 1_000_000_000) {
|
||||||
|
return `${(value / 1_000_000_000).toFixed(2)}B`;
|
||||||
|
} else if (value >= 1_000_000) {
|
||||||
|
return `${(value / 1_000_000).toFixed(2)}M`;
|
||||||
|
} else if (value >= 1_000) {
|
||||||
|
return `${(value / 1_000).toFixed(2)}k`;
|
||||||
|
}
|
||||||
|
return Math.round(value).toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getAttackerSubscript(kill: DetailedKill) {
|
||||||
|
if (kill.npc) {
|
||||||
|
return { label: 'npc', cssClass: 'text-purple-400' };
|
||||||
|
}
|
||||||
|
const count = kill.attacker_count ?? 0;
|
||||||
|
if (count === 1) {
|
||||||
|
return { label: 'solo', cssClass: 'text-green-400' };
|
||||||
|
} else if (count > 1) {
|
||||||
|
return { label: String(count), cssClass: 'text-white' };
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
@@ -0,0 +1,111 @@
|
|||||||
|
const ZKILL_URL = 'https://zkillboard.com';
|
||||||
|
const BASE_IMAGE_URL = 'https://images.evetech.net';
|
||||||
|
|
||||||
|
export function zkillLink(type: 'kill' | 'character' | 'corporation' | 'alliance', id?: number | null): string {
|
||||||
|
if (!id) return `${ZKILL_URL}`;
|
||||||
|
if (type === 'kill') return `${ZKILL_URL}/kill/${id}/`;
|
||||||
|
if (type === 'character') return `${ZKILL_URL}/character/${id}/`;
|
||||||
|
if (type === 'corporation') return `${ZKILL_URL}/corporation/${id}/`;
|
||||||
|
if (type === 'alliance') return `${ZKILL_URL}/alliance/${id}/`;
|
||||||
|
return `${ZKILL_URL}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function eveImageUrl(
|
||||||
|
category: 'characters' | 'corporations' | 'alliances' | 'types',
|
||||||
|
id?: number | null,
|
||||||
|
variation: string = 'icon',
|
||||||
|
size?: number,
|
||||||
|
): string | null {
|
||||||
|
if (!id || id <= 0) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
let url = `${BASE_IMAGE_URL}/${category}/${id}/${variation}`;
|
||||||
|
if (size) {
|
||||||
|
url += `?size=${size}`;
|
||||||
|
}
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildVictimImageUrls(args: {
|
||||||
|
victim_char_id?: number | null;
|
||||||
|
victim_ship_type_id?: number | null;
|
||||||
|
victim_corp_id?: number | null;
|
||||||
|
victim_alliance_id?: number | null;
|
||||||
|
}) {
|
||||||
|
const { victim_char_id, victim_ship_type_id, victim_corp_id, victim_alliance_id } = args;
|
||||||
|
|
||||||
|
const victimPortraitUrl = eveImageUrl('characters', victim_char_id, 'portrait', 64);
|
||||||
|
const victimShipUrl = eveImageUrl('types', victim_ship_type_id, 'render', 64);
|
||||||
|
const victimCorpLogoUrl = eveImageUrl('corporations', victim_corp_id, 'logo', 32);
|
||||||
|
const victimAllianceLogoUrl = eveImageUrl('alliances', victim_alliance_id, 'logo', 32);
|
||||||
|
|
||||||
|
return {
|
||||||
|
victimPortraitUrl,
|
||||||
|
victimShipUrl,
|
||||||
|
victimCorpLogoUrl,
|
||||||
|
victimAllianceLogoUrl,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildAttackerShipUrl(final_blow_ship_type_id?: number | null): string | null {
|
||||||
|
return eveImageUrl('types', final_blow_ship_type_id, 'render', 64);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildAttackerImageUrls(args: {
|
||||||
|
final_blow_char_id?: number | null;
|
||||||
|
final_blow_corp_id?: number | null;
|
||||||
|
final_blow_alliance_id?: number | null;
|
||||||
|
}) {
|
||||||
|
const { final_blow_char_id, final_blow_corp_id, final_blow_alliance_id } = args;
|
||||||
|
|
||||||
|
const attackerPortraitUrl = eveImageUrl('characters', final_blow_char_id, 'portrait', 64);
|
||||||
|
const attackerCorpLogoUrl = eveImageUrl('corporations', final_blow_corp_id, 'logo', 32);
|
||||||
|
const attackerAllianceLogoUrl = eveImageUrl('alliances', final_blow_alliance_id, 'logo', 32);
|
||||||
|
|
||||||
|
return {
|
||||||
|
attackerPortraitUrl,
|
||||||
|
attackerCorpLogoUrl,
|
||||||
|
attackerAllianceLogoUrl,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getPrimaryLogoAndTooltip(
|
||||||
|
allianceUrl: string | null,
|
||||||
|
corpUrl: string | null,
|
||||||
|
allianceName: string,
|
||||||
|
corpName: string,
|
||||||
|
fallback: string,
|
||||||
|
) {
|
||||||
|
let url: string | null = null;
|
||||||
|
let tooltip = '';
|
||||||
|
|
||||||
|
if (allianceUrl) {
|
||||||
|
url = allianceUrl;
|
||||||
|
tooltip = allianceName || fallback;
|
||||||
|
} else if (corpUrl) {
|
||||||
|
url = corpUrl;
|
||||||
|
tooltip = corpName || fallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
return { url, tooltip };
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getAttackerPrimaryImageAndTooltip(
|
||||||
|
isNpc: boolean,
|
||||||
|
allianceUrl: string | null,
|
||||||
|
corpUrl: string | null,
|
||||||
|
allianceName: string,
|
||||||
|
corpName: string,
|
||||||
|
finalBlowShipTypeId: number,
|
||||||
|
npcFallback: string = 'NPC Attacker',
|
||||||
|
) {
|
||||||
|
if (isNpc) {
|
||||||
|
const shipUrl = buildAttackerShipUrl(finalBlowShipTypeId);
|
||||||
|
return {
|
||||||
|
url: shipUrl,
|
||||||
|
tooltip: npcFallback,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return getPrimaryLogoAndTooltip(allianceUrl, corpUrl, allianceName, corpName, 'Attacker');
|
||||||
|
}
|
||||||
@@ -0,0 +1,51 @@
|
|||||||
|
import { useMemo, useCallback } from 'react';
|
||||||
|
import useLocalStorageState from 'use-local-storage-state';
|
||||||
|
|
||||||
|
export interface KillsWidgetSettings {
|
||||||
|
compact: boolean;
|
||||||
|
showAll: boolean;
|
||||||
|
excludedSystems: number[];
|
||||||
|
version: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const DEFAULT_KILLS_WIDGET_SETTINGS: KillsWidgetSettings = {
|
||||||
|
compact: false,
|
||||||
|
showAll: false,
|
||||||
|
excludedSystems: [],
|
||||||
|
version: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
function mergeWithDefaults(settings?: Partial<KillsWidgetSettings>): KillsWidgetSettings {
|
||||||
|
if (!settings) {
|
||||||
|
return DEFAULT_KILLS_WIDGET_SETTINGS;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...DEFAULT_KILLS_WIDGET_SETTINGS,
|
||||||
|
...settings,
|
||||||
|
excludedSystems: Array.isArray(settings.excludedSystems) ? settings.excludedSystems : [],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useKillsWidgetSettings() {
|
||||||
|
const [rawValue, setRawValue] = useLocalStorageState<KillsWidgetSettings | undefined>('kills:widget:settings');
|
||||||
|
|
||||||
|
const value = useMemo<KillsWidgetSettings>(() => {
|
||||||
|
return mergeWithDefaults(rawValue);
|
||||||
|
}, [rawValue]);
|
||||||
|
|
||||||
|
const setValue = useCallback(
|
||||||
|
(newVal: KillsWidgetSettings | ((prev: KillsWidgetSettings) => KillsWidgetSettings)) => {
|
||||||
|
setRawValue(prev => {
|
||||||
|
const mergedPrev = mergeWithDefaults(prev);
|
||||||
|
|
||||||
|
const nextUnmerged = typeof newVal === 'function' ? newVal(mergedPrev) : newVal;
|
||||||
|
|
||||||
|
return mergeWithDefaults(nextUnmerged);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
[setRawValue],
|
||||||
|
);
|
||||||
|
|
||||||
|
return [value, setValue] as const;
|
||||||
|
}
|
||||||
@@ -0,0 +1,178 @@
|
|||||||
|
import { useCallback, useMemo, useState, useEffect, useRef } from 'react';
|
||||||
|
import debounce from 'lodash.debounce';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers';
|
||||||
|
import { DetailedKill } from '@/hooks/Mapper/types/kills';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useKillsWidgetSettings } from './useKillsWidgetSettings';
|
||||||
|
|
||||||
|
interface UseSystemKillsProps {
|
||||||
|
systemId?: string;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
outCommand: (payload: any) => Promise<any>;
|
||||||
|
showAllVisible?: boolean;
|
||||||
|
sinceHours?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
function combineKills(existing: DetailedKill[], incoming: DetailedKill[], sinceHours: number): DetailedKill[] {
|
||||||
|
const cutoff = Date.now() - sinceHours * 60 * 60 * 1000;
|
||||||
|
const byId: Record<string, DetailedKill> = {};
|
||||||
|
|
||||||
|
for (const kill of [...existing, ...incoming]) {
|
||||||
|
if (!kill.kill_time) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const killTimeMs = new Date(kill.kill_time).valueOf();
|
||||||
|
|
||||||
|
if (killTimeMs >= cutoff) {
|
||||||
|
byId[kill.killmail_id] = kill;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Object.values(byId);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useSystemKills({ systemId, outCommand, showAllVisible = false, sinceHours = 24 }: UseSystemKillsProps) {
|
||||||
|
const { data, update } = useMapRootState();
|
||||||
|
const { detailedKills = {}, systems = [] } = data;
|
||||||
|
|
||||||
|
const [settings] = useKillsWidgetSettings();
|
||||||
|
const excludedSystems = settings.excludedSystems;
|
||||||
|
|
||||||
|
const visibleSystemIds = useMemo(() => {
|
||||||
|
return systems.map(s => s.id).filter(id => !excludedSystems.includes(Number(id)));
|
||||||
|
}, [systems, excludedSystems]);
|
||||||
|
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [error, setError] = useState<string | null>(null);
|
||||||
|
|
||||||
|
const didFallbackFetch = useRef(Object.keys(detailedKills).length !== 0);
|
||||||
|
|
||||||
|
const mergeKillsIntoGlobal = useCallback(
|
||||||
|
(killsMap: Record<string, DetailedKill[]>) => {
|
||||||
|
update(prev => {
|
||||||
|
const oldMap = prev.detailedKills ?? {};
|
||||||
|
const updated: Record<string, DetailedKill[]> = { ...oldMap };
|
||||||
|
|
||||||
|
for (const [sid, newKills] of Object.entries(killsMap)) {
|
||||||
|
const existing = updated[sid] ?? [];
|
||||||
|
const combined = combineKills(existing, newKills, sinceHours);
|
||||||
|
updated[sid] = combined;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...prev,
|
||||||
|
detailedKills: updated,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
},
|
||||||
|
[update, sinceHours],
|
||||||
|
);
|
||||||
|
|
||||||
|
const fetchKills = useCallback(
|
||||||
|
async (forceFallback = false) => {
|
||||||
|
setIsLoading(true);
|
||||||
|
setError(null);
|
||||||
|
|
||||||
|
try {
|
||||||
|
let eventType: OutCommand;
|
||||||
|
let requestData: Record<string, unknown>;
|
||||||
|
|
||||||
|
if (showAllVisible || forceFallback) {
|
||||||
|
eventType = OutCommand.getSystemsKills;
|
||||||
|
requestData = {
|
||||||
|
system_ids: visibleSystemIds,
|
||||||
|
since_hours: sinceHours,
|
||||||
|
};
|
||||||
|
} else if (systemId) {
|
||||||
|
eventType = OutCommand.getSystemKills;
|
||||||
|
requestData = {
|
||||||
|
system_id: systemId,
|
||||||
|
since_hours: sinceHours,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
// If there's no system and not showing all, do nothing
|
||||||
|
setIsLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const resp = await outCommand({
|
||||||
|
type: eventType,
|
||||||
|
data: requestData,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Single system => `resp.kills`
|
||||||
|
if (resp.kills) {
|
||||||
|
const arr = resp.kills as DetailedKill[];
|
||||||
|
const sid = systemId ?? 'unknown';
|
||||||
|
mergeKillsIntoGlobal({ [sid]: arr });
|
||||||
|
}
|
||||||
|
// multiple => `resp.systems_kills`
|
||||||
|
else if (resp.systems_kills) {
|
||||||
|
mergeKillsIntoGlobal(resp.systems_kills as Record<string, DetailedKill[]>);
|
||||||
|
} else {
|
||||||
|
console.warn('[useSystemKills] Unexpected kills response =>', resp);
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
console.error('[useSystemKills] Failed to fetch kills:', err);
|
||||||
|
setError(err instanceof Error ? err.message : 'Error fetching kills');
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[showAllVisible, systemId, outCommand, visibleSystemIds, sinceHours, mergeKillsIntoGlobal],
|
||||||
|
);
|
||||||
|
|
||||||
|
const debouncedFetchKills = useMemo(
|
||||||
|
() =>
|
||||||
|
debounce(fetchKills, 500, {
|
||||||
|
leading: true,
|
||||||
|
trailing: false,
|
||||||
|
}),
|
||||||
|
[fetchKills],
|
||||||
|
);
|
||||||
|
|
||||||
|
const finalKills = useMemo(() => {
|
||||||
|
if (showAllVisible) {
|
||||||
|
return visibleSystemIds.flatMap(sid => detailedKills[sid] ?? []);
|
||||||
|
} else if (systemId) {
|
||||||
|
return detailedKills[systemId] ?? [];
|
||||||
|
} else if (didFallbackFetch.current) {
|
||||||
|
// if we already did a fallback, we may have data for multiple systems
|
||||||
|
return visibleSystemIds.flatMap(sid => detailedKills[sid] ?? []);
|
||||||
|
}
|
||||||
|
return [];
|
||||||
|
}, [showAllVisible, systemId, didFallbackFetch, visibleSystemIds, detailedKills]);
|
||||||
|
|
||||||
|
const effectiveIsLoading = isLoading && finalKills.length === 0;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!systemId && !showAllVisible && !didFallbackFetch.current) {
|
||||||
|
didFallbackFetch.current = true;
|
||||||
|
// Cancel any queued debounced calls, then do the fallback.
|
||||||
|
debouncedFetchKills.cancel();
|
||||||
|
fetchKills(true); // forceFallback => fetch as though showAll
|
||||||
|
}
|
||||||
|
}, [systemId, showAllVisible, debouncedFetchKills, fetchKills, didFallbackFetch]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (visibleSystemIds.length === 0) return;
|
||||||
|
|
||||||
|
if (showAllVisible || systemId) {
|
||||||
|
debouncedFetchKills();
|
||||||
|
// Clean up the debounce on unmount or changes
|
||||||
|
return () => debouncedFetchKills.cancel();
|
||||||
|
}
|
||||||
|
}, [showAllVisible, systemId, visibleSystemIds, debouncedFetchKills]);
|
||||||
|
|
||||||
|
const refetch = useCallback(() => {
|
||||||
|
debouncedFetchKills.cancel();
|
||||||
|
fetchKills(); // immediate (non-debounced) call
|
||||||
|
}, [debouncedFetchKills, fetchKills]);
|
||||||
|
|
||||||
|
return {
|
||||||
|
kills: finalKills,
|
||||||
|
isLoading: effectiveIsLoading,
|
||||||
|
error,
|
||||||
|
refetch,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SystemKills';
|
||||||
@@ -2,7 +2,10 @@ import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
|||||||
import { parseSignatures } from '@/hooks/Mapper/helpers';
|
import { parseSignatures } from '@/hooks/Mapper/helpers';
|
||||||
import { Commands, 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 { GROUPS_LIST } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
import {
|
||||||
|
getGroupIdByRawGroup,
|
||||||
|
GROUPS_LIST,
|
||||||
|
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
||||||
|
|
||||||
import { DataTable, DataTableRowClickEvent, DataTableRowMouseEvent, SortOrder } from 'primereact/datatable';
|
import { DataTable, DataTableRowClickEvent, DataTableRowMouseEvent, SortOrder } from 'primereact/datatable';
|
||||||
import { Column } from 'primereact/column';
|
import { Column } from 'primereact/column';
|
||||||
@@ -122,13 +125,14 @@ export const SystemSignaturesContent = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
const isCosmicSignature = x.kind === COSMIC_SIGNATURE;
|
const isCosmicSignature = x.kind === COSMIC_SIGNATURE;
|
||||||
|
const preparedGroup = getGroupIdByRawGroup(x.group);
|
||||||
|
|
||||||
if (isCosmicSignature) {
|
if (isCosmicSignature) {
|
||||||
const showCosmicSignatures = settings.find(y => y.key === COSMIC_SIGNATURE)?.value;
|
const showCosmicSignatures = settings.find(y => y.key === COSMIC_SIGNATURE)?.value;
|
||||||
if (showCosmicSignatures) {
|
if (showCosmicSignatures) {
|
||||||
return !x.group || groupSettings.find(y => y.key === x.group)?.value;
|
return !x.group || groupSettings.find(y => y.key === preparedGroup)?.value;
|
||||||
} else {
|
} else {
|
||||||
return !!x.group && groupSettings.find(y => y.key === x.group)?.value;
|
return !!x.group && groupSettings.find(y => y.key === preparedGroup)?.value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,12 @@
|
|||||||
import { GroupType, SignatureGroup } from '@/hooks/Mapper/types';
|
import {
|
||||||
|
GroupType,
|
||||||
|
SignatureGroup,
|
||||||
|
SignatureGroupENG,
|
||||||
|
SignatureGroupRU,
|
||||||
|
SignatureKind,
|
||||||
|
SignatureKindENG,
|
||||||
|
SignatureKindRU,
|
||||||
|
} from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
export const TIME_ONE_MINUTE = 1000 * 60;
|
export const TIME_ONE_MINUTE = 1000 * 60;
|
||||||
export const TIME_TEN_MINUTES = 1000 * 60 * 10;
|
export const TIME_TEN_MINUTES = 1000 * 60 * 10;
|
||||||
@@ -24,3 +32,43 @@ export const GROUPS: Record<SignatureGroup, GroupType> = {
|
|||||||
[SignatureGroup.Wormhole]: { id: SignatureGroup.Wormhole, icon: '/icons/brackets/wormhole.png', ...wh },
|
[SignatureGroup.Wormhole]: { id: SignatureGroup.Wormhole, icon: '/icons/brackets/wormhole.png', ...wh },
|
||||||
[SignatureGroup.CosmicSignature]: { id: SignatureGroup.CosmicSignature, icon: '/icons/x_close14.png', w: 9, h: 9 },
|
[SignatureGroup.CosmicSignature]: { id: SignatureGroup.CosmicSignature, icon: '/icons/x_close14.png', w: 9, h: 9 },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const MAPPING_GROUP_TO_ENG = {
|
||||||
|
// ENGLISH
|
||||||
|
[SignatureGroupENG.GasSite]: SignatureGroup.GasSite,
|
||||||
|
[SignatureGroupENG.RelicSite]: SignatureGroup.RelicSite,
|
||||||
|
[SignatureGroupENG.DataSite]: SignatureGroup.DataSite,
|
||||||
|
[SignatureGroupENG.OreSite]: SignatureGroup.OreSite,
|
||||||
|
[SignatureGroupENG.CombatSite]: SignatureGroup.CombatSite,
|
||||||
|
[SignatureGroupENG.Wormhole]: SignatureGroup.Wormhole,
|
||||||
|
[SignatureGroupENG.CosmicSignature]: SignatureGroup.CosmicSignature,
|
||||||
|
|
||||||
|
// RUSSIAN
|
||||||
|
[SignatureGroupRU.GasSite]: SignatureGroup.GasSite,
|
||||||
|
[SignatureGroupRU.RelicSite]: SignatureGroup.RelicSite,
|
||||||
|
[SignatureGroupRU.DataSite]: SignatureGroup.DataSite,
|
||||||
|
[SignatureGroupRU.OreSite]: SignatureGroup.OreSite,
|
||||||
|
[SignatureGroupRU.CombatSite]: SignatureGroup.CombatSite,
|
||||||
|
[SignatureGroupRU.Wormhole]: SignatureGroup.Wormhole,
|
||||||
|
[SignatureGroupRU.CosmicSignature]: SignatureGroup.CosmicSignature,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const MAPPING_TYPE_TO_ENG = {
|
||||||
|
// ENGLISH
|
||||||
|
[SignatureKindENG.CosmicSignature]: SignatureKind.CosmicSignature,
|
||||||
|
[SignatureKindENG.CosmicAnomaly]: SignatureKind.CosmicAnomaly,
|
||||||
|
[SignatureKindENG.Structure]: SignatureKind.Structure,
|
||||||
|
[SignatureKindENG.Ship]: SignatureKind.Ship,
|
||||||
|
[SignatureKindENG.Deployable]: SignatureKind.Deployable,
|
||||||
|
[SignatureKindENG.Drone]: SignatureKind.Drone,
|
||||||
|
|
||||||
|
// RUSSIAN
|
||||||
|
[SignatureKindRU.CosmicSignature]: SignatureKind.CosmicSignature,
|
||||||
|
[SignatureKindRU.CosmicAnomaly]: SignatureKind.CosmicAnomaly,
|
||||||
|
[SignatureKindRU.Structure]: SignatureKind.Structure,
|
||||||
|
[SignatureKindRU.Ship]: SignatureKind.Ship,
|
||||||
|
[SignatureKindRU.Deployable]: SignatureKind.Deployable,
|
||||||
|
[SignatureKindRU.Drone]: SignatureKind.Drone,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getGroupIdByRawGroup = (val: string) => MAPPING_GROUP_TO_ENG[val as SignatureGroup];
|
||||||
|
|||||||
@@ -1,28 +1,29 @@
|
|||||||
import { PrimeIcons } from 'primereact/api';
|
import { PrimeIcons } from 'primereact/api';
|
||||||
import { SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
import { SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
import { SystemViewStandalone, WHClassView } from '@/hooks/Mapper/components/ui-kit';
|
import { SystemViewStandalone, TooltipPosition, WHClassView } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
import {
|
import { renderK162Type } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureK162TypeSelect';
|
||||||
k162Types,
|
|
||||||
renderK162Type,
|
|
||||||
} from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureK162TypeSelect';
|
|
||||||
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
||||||
|
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import { renderName } from './renderName.tsx';
|
import { renderName } from './renderName.tsx';
|
||||||
|
import { K162_TYPES_MAP } from '@/hooks/Mapper/constants.ts';
|
||||||
|
import { parseSignatureCustomInfo } from '@/hooks/Mapper/helpers/parseSignatureCustomInfo.ts';
|
||||||
|
|
||||||
export const renderInfoColumn = (row: SystemSignature) => {
|
export const renderInfoColumn = (row: SystemSignature) => {
|
||||||
if (!row.group || row.group === SignatureGroup.Wormhole) {
|
if (!row.group || row.group === SignatureGroup.Wormhole) {
|
||||||
let k162TypeOption = null;
|
const customInfo = parseSignatureCustomInfo(row.custom_info);
|
||||||
if (row.custom_info) {
|
|
||||||
const customInfo = JSON.parse(row.custom_info);
|
const k162TypeOption = customInfo.k162Type ? K162_TYPES_MAP[customInfo.k162Type] : null;
|
||||||
if (customInfo.k162Type) {
|
|
||||||
k162TypeOption = k162Types.find(x => x.value === customInfo.k162Type);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex justify-start items-center gap-[4px]">
|
<div className="flex justify-start items-center gap-[4px]">
|
||||||
|
{customInfo.isEOL && (
|
||||||
|
<WdTooltipWrapper offset={5} position={TooltipPosition.top} content="Signature marked as EOL">
|
||||||
|
<div className="pi pi-clock text-fuchsia-400 text-[11px] mr-[2px]"></div>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
)}
|
||||||
|
|
||||||
{row.type && (
|
{row.type && (
|
||||||
<WHClassView
|
<WHClassView
|
||||||
className="text-[11px]"
|
className="text-[11px]"
|
||||||
@@ -34,7 +35,7 @@ export const renderInfoColumn = (row: SystemSignature) => {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{!row.linked_system && row.type === 'K162' && !!k162TypeOption && <>{renderK162Type(k162TypeOption)}</>}
|
{!row.linked_system && row.type === 'K162' && k162TypeOption && renderK162Type(k162TypeOption)}
|
||||||
|
|
||||||
{row.linked_system && (
|
{row.linked_system && (
|
||||||
<>
|
<>
|
||||||
|
|||||||
@@ -0,0 +1,118 @@
|
|||||||
|
import React, { useCallback, ClipboardEvent, useRef } from 'react';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import useMaxWidth from '@/hooks/Mapper/hooks/useMaxWidth';
|
||||||
|
import {
|
||||||
|
LayoutEventBlocker,
|
||||||
|
WdImgButton,
|
||||||
|
TooltipPosition,
|
||||||
|
InfoDrawer,
|
||||||
|
SystemView,
|
||||||
|
} from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { Widget } from '@/hooks/Mapper/components/mapInterface/components';
|
||||||
|
|
||||||
|
import { SystemStructuresContent } from './SystemStructuresContent/SystemStructuresContent';
|
||||||
|
import { useSystemStructures } from './hooks/useSystemStructures';
|
||||||
|
import { processSnippetText } from './helpers';
|
||||||
|
|
||||||
|
export const SystemStructures: React.FC = () => {
|
||||||
|
const {
|
||||||
|
data: { selectedSystems },
|
||||||
|
outCommand,
|
||||||
|
} = useMapRootState();
|
||||||
|
const [systemId] = selectedSystems;
|
||||||
|
const isNotSelectedSystem = selectedSystems.length !== 1;
|
||||||
|
|
||||||
|
const { structures, handleUpdateStructures } = useSystemStructures({ systemId, outCommand });
|
||||||
|
|
||||||
|
const labelRef = useRef<HTMLDivElement>(null);
|
||||||
|
const isCompact = useMaxWidth(labelRef, 260);
|
||||||
|
|
||||||
|
const processClipboard = useCallback(
|
||||||
|
(text: string) => {
|
||||||
|
const updated = processSnippetText(text, structures);
|
||||||
|
handleUpdateStructures(updated);
|
||||||
|
},
|
||||||
|
[structures, handleUpdateStructures],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handlePaste = useCallback(
|
||||||
|
(e: ClipboardEvent<HTMLDivElement>) => {
|
||||||
|
e.preventDefault();
|
||||||
|
processClipboard(e.clipboardData.getData('text'));
|
||||||
|
},
|
||||||
|
[processClipboard],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handlePasteTimer = useCallback(async () => {
|
||||||
|
try {
|
||||||
|
const text = await navigator.clipboard.readText();
|
||||||
|
processClipboard(text);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Clipboard read error:', err);
|
||||||
|
}
|
||||||
|
}, [processClipboard]);
|
||||||
|
|
||||||
|
function renderWidgetLabel() {
|
||||||
|
return (
|
||||||
|
<div className="flex justify-between items-center text-xs w-full h-full" ref={labelRef}>
|
||||||
|
<div className="flex justify-between items-center gap-1">
|
||||||
|
{!isCompact && (
|
||||||
|
<div className="flex whitespace-nowrap text-ellipsis overflow-hidden text-stone-400">
|
||||||
|
Structures
|
||||||
|
{!isNotSelectedSystem && ' in'}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{!isNotSelectedSystem && <SystemView systemId={systemId} className="select-none text-center" hideRegion />}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<LayoutEventBlocker className="flex gap-2.5">
|
||||||
|
<WdImgButton
|
||||||
|
className={`${PrimeIcons.CLOCK} text-sky-400 hover:text-sky-200 transition duration-300`}
|
||||||
|
onClick={handlePasteTimer}
|
||||||
|
tooltip={{
|
||||||
|
position: TooltipPosition.left,
|
||||||
|
// @ts-ignore
|
||||||
|
content: 'Add Structures/Timer',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<WdImgButton
|
||||||
|
className={PrimeIcons.QUESTION_CIRCLE}
|
||||||
|
tooltip={{
|
||||||
|
position: TooltipPosition.left,
|
||||||
|
// @ts-ignore
|
||||||
|
content: (
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<InfoDrawer title={<b className="text-slate-50">How to add/update structures?</b>}>
|
||||||
|
In game, select one or more structures in D-Scan and then
|
||||||
|
<br />
|
||||||
|
use the blue add structure data button
|
||||||
|
</InfoDrawer>
|
||||||
|
<InfoDrawer title={<b className="text-slate-50">How to add a timer?</b>}>
|
||||||
|
In game, select a structure with an active timer, right click to copy, and then
|
||||||
|
<span className="text-blue-500"> blue </span>
|
||||||
|
use the blue add structure data button
|
||||||
|
</InfoDrawer>
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</LayoutEventBlocker>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div tabIndex={0} onPaste={handlePaste} className="h-full flex flex-col" style={{ outline: 'none' }}>
|
||||||
|
<Widget label={renderWidgetLabel()}>
|
||||||
|
{isNotSelectedSystem ? (
|
||||||
|
<div className="w-full h-full flex justify-center items-center select-none text-center text-stone-400/80 text-sm">
|
||||||
|
System is not selected
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<SystemStructuresContent structures={structures} onUpdateStructures={handleUpdateStructures} />
|
||||||
|
)}
|
||||||
|
</Widget>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
.TableRowCompact {
|
||||||
|
height: 8px;
|
||||||
|
max-height: 8px;
|
||||||
|
font-size: 12px !important;
|
||||||
|
line-height: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.Table {
|
||||||
|
font-size: 12px;
|
||||||
|
border-collapse: collapse;
|
||||||
|
table-layout: fixed;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.Table .p-datatable-tbody > tr > td {
|
||||||
|
white-space: nowrap;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.Tooltip {
|
||||||
|
white-space: pre-line;
|
||||||
|
line-height: 1.2rem;
|
||||||
|
}
|
||||||
@@ -0,0 +1,170 @@
|
|||||||
|
import React, { useState, useCallback, useMemo } from 'react';
|
||||||
|
import { DataTable, DataTableRowClickEvent } from 'primereact/datatable';
|
||||||
|
import { Column } from 'primereact/column';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
|
||||||
|
import { SystemStructuresDialog } from '../SystemStructuresDialog/SystemStructuresDialog';
|
||||||
|
import { StructureItem } from '../helpers/structureTypes';
|
||||||
|
import { useHotkey } from '@/hooks/Mapper/hooks';
|
||||||
|
import classes from './SystemStructuresContent.module.scss';
|
||||||
|
import { renderOwnerCell, renderTypeCell, renderTimerCell } from '../renders/cellRenders';
|
||||||
|
|
||||||
|
interface SystemStructuresContentProps {
|
||||||
|
structures: StructureItem[];
|
||||||
|
onUpdateStructures: (newList: StructureItem[]) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SystemStructuresContent: React.FC<SystemStructuresContentProps> = ({ structures, onUpdateStructures }) => {
|
||||||
|
const [selectedRow, setSelectedRow] = useState<StructureItem | null>(null);
|
||||||
|
const [editingItem, setEditingItem] = useState<StructureItem | null>(null);
|
||||||
|
const [showEditDialog, setShowEditDialog] = useState(false);
|
||||||
|
|
||||||
|
const handleRowClick = (e: DataTableRowClickEvent) => {
|
||||||
|
const row = e.data as StructureItem;
|
||||||
|
setSelectedRow(prev => (prev?.id === row.id ? null : row));
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRowDoubleClick = (e: DataTableRowClickEvent) => {
|
||||||
|
setEditingItem(e.data as StructureItem);
|
||||||
|
setShowEditDialog(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Press Delete => remove selected row
|
||||||
|
const handleDeleteSelected = useCallback(
|
||||||
|
(e: KeyboardEvent) => {
|
||||||
|
if (!selectedRow) return;
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
|
||||||
|
const newList = structures.filter(s => s.id !== selectedRow.id);
|
||||||
|
onUpdateStructures(newList);
|
||||||
|
setSelectedRow(null);
|
||||||
|
},
|
||||||
|
[selectedRow, structures, onUpdateStructures],
|
||||||
|
);
|
||||||
|
|
||||||
|
useHotkey(false, ['Delete', 'Backspace'], handleDeleteSelected);
|
||||||
|
|
||||||
|
const visibleStructures = useMemo(() => {
|
||||||
|
return structures;
|
||||||
|
}, [structures]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col gap-2 p-2 text-xs text-stone-200 h-full">
|
||||||
|
{visibleStructures.length === 0 ? (
|
||||||
|
<div className="flex-1 flex justify-center items-center text-stone-400/80 text-sm">No structures</div>
|
||||||
|
) : (
|
||||||
|
<div className="flex-1">
|
||||||
|
<DataTable
|
||||||
|
value={visibleStructures}
|
||||||
|
dataKey="id"
|
||||||
|
className={clsx(classes.Table, 'w-full select-none h-full')}
|
||||||
|
size="small"
|
||||||
|
sortMode="single"
|
||||||
|
rowHover
|
||||||
|
style={{ tableLayout: 'fixed', width: '100%' }}
|
||||||
|
onRowClick={handleRowClick}
|
||||||
|
onRowDoubleClick={handleRowDoubleClick}
|
||||||
|
rowClassName={rowData => {
|
||||||
|
const isSelected = selectedRow?.id === rowData.id;
|
||||||
|
return clsx(
|
||||||
|
classes.TableRowCompact,
|
||||||
|
'transition-colors duration-200 cursor-pointer',
|
||||||
|
isSelected ? 'bg-amber-500/50 hover:bg-amber-500/70' : 'hover:bg-purple-400/20',
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Column
|
||||||
|
header="Type"
|
||||||
|
body={renderTypeCell}
|
||||||
|
style={{
|
||||||
|
width: '160px',
|
||||||
|
whiteSpace: 'nowrap',
|
||||||
|
overflow: 'hidden',
|
||||||
|
textOverflow: 'ellipsis',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Column
|
||||||
|
field="name"
|
||||||
|
header="Name"
|
||||||
|
style={{
|
||||||
|
width: '120px',
|
||||||
|
whiteSpace: 'nowrap',
|
||||||
|
overflow: 'hidden',
|
||||||
|
textOverflow: 'ellipsis',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Column
|
||||||
|
header="Owner"
|
||||||
|
body={renderOwnerCell}
|
||||||
|
style={{
|
||||||
|
width: '120px',
|
||||||
|
whiteSpace: 'nowrap',
|
||||||
|
overflow: 'hidden',
|
||||||
|
textOverflow: 'ellipsis',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Column
|
||||||
|
field="status"
|
||||||
|
header="Status"
|
||||||
|
style={{
|
||||||
|
width: '100px',
|
||||||
|
whiteSpace: 'nowrap',
|
||||||
|
overflow: 'hidden',
|
||||||
|
textOverflow: 'ellipsis',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Column
|
||||||
|
header="Timer"
|
||||||
|
body={renderTimerCell}
|
||||||
|
style={{
|
||||||
|
width: '110px',
|
||||||
|
whiteSpace: 'nowrap',
|
||||||
|
overflow: 'hidden',
|
||||||
|
textOverflow: 'ellipsis',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Column
|
||||||
|
body={(rowData: StructureItem) => (
|
||||||
|
<i
|
||||||
|
className={clsx(PrimeIcons.PENCIL, 'text-[14px] cursor-pointer')}
|
||||||
|
title="Edit"
|
||||||
|
onClick={() => {
|
||||||
|
setEditingItem(rowData);
|
||||||
|
setShowEditDialog(true);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
style={{
|
||||||
|
width: '40px',
|
||||||
|
textAlign: 'center',
|
||||||
|
whiteSpace: 'nowrap',
|
||||||
|
overflow: 'hidden',
|
||||||
|
textOverflow: 'ellipsis',
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</DataTable>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{showEditDialog && editingItem && (
|
||||||
|
<SystemStructuresDialog
|
||||||
|
visible={showEditDialog}
|
||||||
|
structure={editingItem}
|
||||||
|
onClose={() => setShowEditDialog(false)}
|
||||||
|
onSave={(updatedItem: StructureItem) => {
|
||||||
|
const newList = structures.map(s => (s.id === updatedItem.id ? updatedItem : s));
|
||||||
|
onUpdateStructures(newList);
|
||||||
|
setShowEditDialog(false);
|
||||||
|
}}
|
||||||
|
onDelete={(deleteId: string) => {
|
||||||
|
const newList = structures.filter(s => s.id !== deleteId);
|
||||||
|
onUpdateStructures(newList);
|
||||||
|
setShowEditDialog(false);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,31 @@
|
|||||||
|
|
||||||
|
.systemStructureDialog {
|
||||||
|
|
||||||
|
.p-dialog-content {
|
||||||
|
background-color: var(--surface-800) !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dialog-header {
|
||||||
|
background-color: var(--surface-700);
|
||||||
|
color: var(--text-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dialog-header-icon,
|
||||||
|
.p-dialog-header-title {
|
||||||
|
color: var(--gray-200);
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-inputtext {
|
||||||
|
background-color: #2a2a2a !important;
|
||||||
|
color: #ddd !important;
|
||||||
|
font-size: 12px !important;
|
||||||
|
padding: 0.25rem 0.5rem !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-dialog-footer {
|
||||||
|
.p-button {
|
||||||
|
font-size: 12px !important;
|
||||||
|
padding: 0.3rem 0.75rem !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,235 @@
|
|||||||
|
import React, { useEffect, useState, useCallback } from 'react';
|
||||||
|
import { Dialog } from 'primereact/dialog';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { AutoComplete } from 'primereact/autocomplete';
|
||||||
|
import { Calendar } from 'primereact/calendar';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
|
||||||
|
import { StructureItem, StructureStatus, statusesRequiringTimer, formatToISO } from '../helpers';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
|
interface StructuresEditDialogProps {
|
||||||
|
visible: boolean;
|
||||||
|
structure?: StructureItem;
|
||||||
|
onClose: () => void;
|
||||||
|
onSave: (updatedItem: StructureItem) => void;
|
||||||
|
onDelete: (id: string) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SystemStructuresDialog: React.FC<StructuresEditDialogProps> = ({
|
||||||
|
visible,
|
||||||
|
structure,
|
||||||
|
onClose,
|
||||||
|
onSave,
|
||||||
|
onDelete,
|
||||||
|
}) => {
|
||||||
|
const [editData, setEditData] = useState<StructureItem | null>(null);
|
||||||
|
const [ownerInput, setOwnerInput] = useState('');
|
||||||
|
const [ownerSuggestions, setOwnerSuggestions] = useState<{ label: string; value: string }[]>([]);
|
||||||
|
|
||||||
|
const { outCommand } = useMapRootState();
|
||||||
|
|
||||||
|
const [prevQuery, setPrevQuery] = useState('');
|
||||||
|
const [prevResults, setPrevResults] = useState<{ label: string; value: string }[]>([]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (structure) {
|
||||||
|
setEditData(structure);
|
||||||
|
setOwnerInput(structure.ownerName ?? '');
|
||||||
|
} else {
|
||||||
|
setEditData(null);
|
||||||
|
setOwnerInput('');
|
||||||
|
}
|
||||||
|
}, [structure]);
|
||||||
|
|
||||||
|
// Searching corporation owners via auto-complete
|
||||||
|
const searchOwners = useCallback(
|
||||||
|
async (e: { query: string }) => {
|
||||||
|
const newQuery = e.query.trim();
|
||||||
|
if (!newQuery) {
|
||||||
|
setOwnerSuggestions([]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If user typed more text but we have partial match in prevResults
|
||||||
|
if (newQuery.startsWith(prevQuery) && prevResults.length > 0) {
|
||||||
|
const filtered = prevResults.filter(item =>
|
||||||
|
item.label.toLowerCase().includes(newQuery.toLowerCase()),
|
||||||
|
);
|
||||||
|
setOwnerSuggestions(filtered);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const { results = [] } = await outCommand({
|
||||||
|
type: OutCommand.getCorporationNames,
|
||||||
|
data: { search: newQuery },
|
||||||
|
});
|
||||||
|
setOwnerSuggestions(results);
|
||||||
|
setPrevQuery(newQuery);
|
||||||
|
setPrevResults(results);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to fetch owners:', err);
|
||||||
|
setOwnerSuggestions([]);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[prevQuery, prevResults, outCommand],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleChange = (field: keyof StructureItem, val: string | Date) => {
|
||||||
|
// If we want to forbid changing structureTypeId or structureType from the dialog, do so here:
|
||||||
|
if (field === 'structureTypeId' || field === 'structureType') return;
|
||||||
|
|
||||||
|
setEditData(prev => {
|
||||||
|
if (!prev) return null;
|
||||||
|
|
||||||
|
// If this is the endTime (Date from Calendar), we store as ISO or string:
|
||||||
|
if (field === 'endTime' && val instanceof Date) {
|
||||||
|
return { ...prev, endTime: val.toISOString() };
|
||||||
|
}
|
||||||
|
|
||||||
|
return { ...prev, [field]: val };
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// when user picks a corp from auto-complete
|
||||||
|
const handleSelectOwner = (selected: { label: string; value: string }) => {
|
||||||
|
setOwnerInput(selected.label);
|
||||||
|
setEditData(prev =>
|
||||||
|
prev ? { ...prev, ownerName: selected.label, ownerId: selected.value } : null,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleStatusChange = (val: string) => {
|
||||||
|
setEditData(prev => {
|
||||||
|
if (!prev) return null;
|
||||||
|
const newStatus = val as StructureStatus;
|
||||||
|
// If new status doesn't require a timer, we clear out endTime
|
||||||
|
const newEndTime = statusesRequiringTimer.includes(newStatus) ? prev.endTime : '';
|
||||||
|
return { ...prev, status: newStatus, endTime: newEndTime };
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSaveClick = async () => {
|
||||||
|
if (!editData) return;
|
||||||
|
|
||||||
|
// If status doesn't require a timer, clear endTime
|
||||||
|
if (!statusesRequiringTimer.includes(editData.status)) {
|
||||||
|
editData.endTime = '';
|
||||||
|
} else if (editData.endTime) {
|
||||||
|
// convert to full ISO if not already
|
||||||
|
editData.endTime = formatToISO(editData.endTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
// fetch corporation ticker if we have an ownerId
|
||||||
|
if (editData.ownerId) {
|
||||||
|
try {
|
||||||
|
const { ticker } = await outCommand({
|
||||||
|
type: OutCommand.getCorporationTicker,
|
||||||
|
data: { corp_id: editData.ownerId },
|
||||||
|
});
|
||||||
|
editData.ownerTicker = ticker ?? '';
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to fetch ticker:', err);
|
||||||
|
editData.ownerTicker = '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onSave(editData);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteClick = () => {
|
||||||
|
if (!editData) return;
|
||||||
|
onDelete(editData.id);
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!editData) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Dialog
|
||||||
|
visible={visible}
|
||||||
|
onHide={onClose}
|
||||||
|
header={`Edit Structure - ${editData.name ?? ''}`}
|
||||||
|
className={clsx('myStructuresDialog', 'text-stone-200 w-full max-w-md')}
|
||||||
|
>
|
||||||
|
<div className="flex flex-col gap-2 text-[14px]">
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center">
|
||||||
|
<span>Type:</span>
|
||||||
|
<input
|
||||||
|
readOnly
|
||||||
|
className="p-inputtext p-component cursor-not-allowed"
|
||||||
|
value={editData.structureType ?? ''}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center">
|
||||||
|
<span>Name:</span>
|
||||||
|
<input
|
||||||
|
className="p-inputtext p-component"
|
||||||
|
value={editData.name ?? ''}
|
||||||
|
onChange={e => handleChange('name', e.target.value)}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center">
|
||||||
|
<span>Owner:</span>
|
||||||
|
<AutoComplete
|
||||||
|
id="owner"
|
||||||
|
value={ownerInput}
|
||||||
|
suggestions={ownerSuggestions}
|
||||||
|
completeMethod={searchOwners}
|
||||||
|
minLength={3}
|
||||||
|
delay={400}
|
||||||
|
field="label"
|
||||||
|
placeholder="Corporation name..."
|
||||||
|
onChange={e => setOwnerInput(e.value)}
|
||||||
|
onSelect={e => handleSelectOwner(e.value)}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center">
|
||||||
|
<span>Status:</span>
|
||||||
|
<select
|
||||||
|
className="p-inputtext p-component"
|
||||||
|
value={editData.status}
|
||||||
|
onChange={e => handleStatusChange(e.target.value)}
|
||||||
|
>
|
||||||
|
<option value="Powered">Powered</option>
|
||||||
|
<option value="Anchoring">Anchoring</option>
|
||||||
|
<option value="Unanchoring">Unanchoring</option>
|
||||||
|
<option value="Low Power">Low Power</option>
|
||||||
|
<option value="Abandoned">Abandoned</option>
|
||||||
|
<option value="Reinforced">Reinforced</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
|
||||||
|
{statusesRequiringTimer.includes(editData.status) && (
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center">
|
||||||
|
<span>Timer <br /> (Eve Time):</span>
|
||||||
|
<Calendar
|
||||||
|
value={editData.endTime ? new Date(editData.endTime) : undefined}
|
||||||
|
onChange={(e) => handleChange('endTime', e.value ?? '')}
|
||||||
|
showTime
|
||||||
|
hourFormat="24"
|
||||||
|
dateFormat="yy-mm-dd"
|
||||||
|
showIcon
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<label className="grid grid-cols-[100px_1fr] gap-2 items-start mt-2">
|
||||||
|
<span className="mt-1">Notes:</span>
|
||||||
|
<textarea
|
||||||
|
className="p-inputtext p-component resize-none h-24"
|
||||||
|
value={editData.notes ?? ''}
|
||||||
|
onChange={e => handleChange('notes', e.target.value)}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex justify-end items-center gap-2 mt-4">
|
||||||
|
<Button label="Delete" severity="danger" className="p-button-sm" onClick={handleDeleteClick} />
|
||||||
|
<Button label="Save" className="p-button-sm" onClick={handleSaveClick} />
|
||||||
|
</div>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,4 @@
|
|||||||
|
export * from './parserHelper';
|
||||||
|
export * from './pasteParser';
|
||||||
|
export * from './structureTypes';
|
||||||
|
export * from './structureUtils';
|
||||||
@@ -0,0 +1,92 @@
|
|||||||
|
import { StructureStatus, StructureItem, STRUCTURE_TYPE_MAP } from './structureTypes';
|
||||||
|
import { formatToISO } from './structureUtils';
|
||||||
|
|
||||||
|
// Up to you if you'd like to keep a separate constant here or not
|
||||||
|
export const statusesRequiringTimer: StructureStatus[] = ['Anchoring', 'Reinforced'];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parseFormatOneLine(line):
|
||||||
|
* - Splits by tabs
|
||||||
|
* - First col => structureTypeId
|
||||||
|
* - Second col => rawName
|
||||||
|
* - Third col => structureTypeName
|
||||||
|
*/
|
||||||
|
export function parseFormatOneLine(line: string): StructureItem | null {
|
||||||
|
const columns = line
|
||||||
|
.split('\t')
|
||||||
|
.map(c => c.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
// Expecting e.g. "35832 J214811 - SomeName Astrahus"
|
||||||
|
if (columns.length < 3) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const [rawTypeId, rawName, rawTypeName] = columns;
|
||||||
|
|
||||||
|
if (columns.length != 4) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!STRUCTURE_TYPE_MAP[rawTypeId]) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rawTypeName != STRUCTURE_TYPE_MAP[rawTypeId]) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const name = rawName.replace(/^J\d{6}\s*-\s*/, '').trim();
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: crypto.randomUUID(),
|
||||||
|
structureTypeId: rawTypeId,
|
||||||
|
structureType: rawTypeName,
|
||||||
|
name,
|
||||||
|
ownerName: '',
|
||||||
|
notes: '',
|
||||||
|
status: 'Powered', // Default
|
||||||
|
endTime: '', // No timer by default
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function matchesThreeLineSnippet(lines: string[]): boolean {
|
||||||
|
if (lines.length < 3) return false;
|
||||||
|
return /until\s+\d{4}\.\d{2}\.\d{2}/i.test(lines[2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* parseThreeLineSnippet:
|
||||||
|
* - Example lines:
|
||||||
|
* line1: "J214811 - Folgers"
|
||||||
|
* line2: "1,475 km"
|
||||||
|
* line3: "Reinforced until 2025.01.13 23:51"
|
||||||
|
*/
|
||||||
|
export function parseThreeLineSnippet(lines: string[]): StructureItem {
|
||||||
|
const [line1, , line3] = lines;
|
||||||
|
|
||||||
|
let status: StructureStatus = 'Reinforced';
|
||||||
|
let endTime: string | undefined;
|
||||||
|
|
||||||
|
// e.g. "Reinforced until 2025.01.13 23:27"
|
||||||
|
const match = line3.match(/^(?<stat>\w+)\s+until\s+(?<dateTime>[\d.]+\s+[\d:]+)/i);
|
||||||
|
|
||||||
|
if (match?.groups?.stat) {
|
||||||
|
const candidateStatus = match.groups.stat as StructureStatus;
|
||||||
|
if (statusesRequiringTimer.includes(candidateStatus)) {
|
||||||
|
status = candidateStatus;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (match?.groups?.dateTime) {
|
||||||
|
let dt = match.groups.dateTime.trim().replace(/\./g, '-'); // "2025-01-13 23:27"
|
||||||
|
dt = dt.replace(' ', 'T'); // "2025-01-13T23:27"
|
||||||
|
endTime = formatToISO(dt); // => "2025-01-13T23:27:00Z"
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: crypto.randomUUID(),
|
||||||
|
name: line1.replace(/^J\d{6}\s*-\s*/, '').trim(),
|
||||||
|
status,
|
||||||
|
endTime,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,56 @@
|
|||||||
|
import { StructureItem } from './structureTypes';
|
||||||
|
import { parseThreeLineSnippet, parseFormatOneLine, matchesThreeLineSnippet } from './parserHelper';
|
||||||
|
|
||||||
|
export function processSnippetText(rawText: string, existingStructures: StructureItem[]): StructureItem[] {
|
||||||
|
if (!rawText) {
|
||||||
|
return existingStructures.slice();
|
||||||
|
}
|
||||||
|
|
||||||
|
const lines = rawText
|
||||||
|
.split(/\r?\n/)
|
||||||
|
.map(line => line.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
if (lines.length === 3 && matchesThreeLineSnippet(lines)) {
|
||||||
|
return applyThreeLineSnippet(lines, existingStructures);
|
||||||
|
} else {
|
||||||
|
return applySingleLineParse(lines, existingStructures);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function applyThreeLineSnippet(snippetLines: string[], existingStructures: StructureItem[]): StructureItem[] {
|
||||||
|
const updatedList = [...existingStructures];
|
||||||
|
const snippetItem = parseThreeLineSnippet(snippetLines);
|
||||||
|
|
||||||
|
const existingIndex = updatedList.findIndex(s => s.name.trim() === snippetItem.name.trim());
|
||||||
|
|
||||||
|
if (existingIndex !== -1) {
|
||||||
|
const existing = updatedList[existingIndex];
|
||||||
|
updatedList[existingIndex] = {
|
||||||
|
...existing,
|
||||||
|
status: snippetItem.status,
|
||||||
|
endTime: snippetItem.endTime,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return updatedList;
|
||||||
|
}
|
||||||
|
|
||||||
|
function applySingleLineParse(lines: string[], existingStructures: StructureItem[]): StructureItem[] {
|
||||||
|
const updatedList = [...existingStructures];
|
||||||
|
const newItems: StructureItem[] = [];
|
||||||
|
|
||||||
|
for (const line of lines) {
|
||||||
|
const item = parseFormatOneLine(line);
|
||||||
|
if (!item) continue;
|
||||||
|
|
||||||
|
const isDuplicate = updatedList.some(
|
||||||
|
s => s.structureTypeId === item.structureTypeId && s.name.trim() === item.name.trim(),
|
||||||
|
);
|
||||||
|
if (!isDuplicate) {
|
||||||
|
newItems.push(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return [...updatedList, ...newItems];
|
||||||
|
}
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
export type StructureStatus = 'Powered' | 'Anchoring' | 'Unanchoring' | 'Low Power' | 'Abandoned' | 'Reinforced';
|
||||||
|
|
||||||
|
export interface StructureItem {
|
||||||
|
id: string;
|
||||||
|
systemId?: string;
|
||||||
|
structureTypeId?: string;
|
||||||
|
structureType?: string;
|
||||||
|
name: string;
|
||||||
|
ownerName?: string;
|
||||||
|
ownerId?: string;
|
||||||
|
ownerTicker?: string;
|
||||||
|
notes?: string;
|
||||||
|
status: StructureStatus;
|
||||||
|
endTime?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const STRUCTURE_TYPE_MAP: Record<string, string> = {
|
||||||
|
'35825': 'Raitaru',
|
||||||
|
'35826': 'Azbel',
|
||||||
|
'35827': 'Sotiyo',
|
||||||
|
'35832': 'Astrahus',
|
||||||
|
'35833': 'Fortizar',
|
||||||
|
'35834': 'Keepstar',
|
||||||
|
'35835': 'Athanor',
|
||||||
|
'35836': 'Tatara',
|
||||||
|
'40340': 'Upwell Palatine Keepstar',
|
||||||
|
'47512': "'Moreau' Fortizar",
|
||||||
|
'47513': "'Draccous' Fortizar",
|
||||||
|
'47514': "'Horizon' Fortizar",
|
||||||
|
'47515': "'Marginis' Fortizar",
|
||||||
|
'47516': "'Prometheus' Fortizar",
|
||||||
|
};
|
||||||
@@ -0,0 +1,59 @@
|
|||||||
|
import { StructureItem } from './structureTypes';
|
||||||
|
|
||||||
|
export function getActualStructures(oldList: StructureItem[], newList: StructureItem[]) {
|
||||||
|
const oldMap = new Map(oldList.map(s => [s.id, s]));
|
||||||
|
const newMap = new Map(newList.map(s => [s.id, s]));
|
||||||
|
|
||||||
|
const added: StructureItem[] = [];
|
||||||
|
const updated: StructureItem[] = [];
|
||||||
|
const removed: StructureItem[] = [];
|
||||||
|
|
||||||
|
for (const newItem of newList) {
|
||||||
|
const oldItem = oldMap.get(newItem.id);
|
||||||
|
if (!oldItem) {
|
||||||
|
added.push(newItem);
|
||||||
|
} else if (JSON.stringify(oldItem) !== JSON.stringify(newItem)) {
|
||||||
|
updated.push(newItem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const oldItem of oldList) {
|
||||||
|
if (!newMap.has(oldItem.id)) {
|
||||||
|
removed.push(oldItem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { added, updated, removed };
|
||||||
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
export function mapServerStructure(serverData: any): StructureItem {
|
||||||
|
const { owner_id, owner_ticker, structure_type_id, structure_type, owner_name, end_time, system_id, ...rest } =
|
||||||
|
serverData;
|
||||||
|
|
||||||
|
return {
|
||||||
|
...rest,
|
||||||
|
ownerId: owner_id,
|
||||||
|
ownerTicker: owner_ticker,
|
||||||
|
ownerName: owner_name,
|
||||||
|
structureType: structure_type,
|
||||||
|
structureTypeId: structure_type_id,
|
||||||
|
endTime: end_time ?? '',
|
||||||
|
systemId: system_id,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function formatToISO(datetimeLocal: string): string {
|
||||||
|
if (!datetimeLocal) return '';
|
||||||
|
|
||||||
|
// If missing seconds, add :00
|
||||||
|
let iso = datetimeLocal;
|
||||||
|
if (/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}$/.test(iso)) {
|
||||||
|
iso += ':00';
|
||||||
|
}
|
||||||
|
// Ensure trailing 'Z'
|
||||||
|
if (!iso.endsWith('Z')) {
|
||||||
|
iso += 'Z';
|
||||||
|
}
|
||||||
|
return iso;
|
||||||
|
}
|
||||||
@@ -0,0 +1,85 @@
|
|||||||
|
import { useEffect, useState, useCallback } from 'react';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers';
|
||||||
|
import { mapServerStructure, getActualStructures, StructureItem, statusesRequiringTimer } from '../helpers';
|
||||||
|
|
||||||
|
interface UseSystemStructuresProps {
|
||||||
|
systemId: string | undefined;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
||||||
|
outCommand: (payload: any) => Promise<any>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useSystemStructures({ systemId, outCommand }: UseSystemStructuresProps) {
|
||||||
|
const [structures, setStructures] = useState<StructureItem[]>([]);
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [error, setError] = useState<string | null>(null);
|
||||||
|
|
||||||
|
const fetchStructures = useCallback(async () => {
|
||||||
|
if (!systemId) {
|
||||||
|
setStructures([]);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setIsLoading(true);
|
||||||
|
setError(null);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const { structures: fetched = [] } = await outCommand({
|
||||||
|
type: OutCommand.getStructures,
|
||||||
|
data: { system_id: systemId },
|
||||||
|
});
|
||||||
|
|
||||||
|
const mappedStructures = fetched.map(mapServerStructure);
|
||||||
|
setStructures(mappedStructures);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to get structures:', err);
|
||||||
|
setError('Error fetching structures');
|
||||||
|
} finally {
|
||||||
|
setIsLoading(false);
|
||||||
|
}
|
||||||
|
}, [systemId, outCommand]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
fetchStructures();
|
||||||
|
}, [fetchStructures]);
|
||||||
|
|
||||||
|
const sanitizeEndTimers = useCallback((item: StructureItem) => {
|
||||||
|
if (!statusesRequiringTimer.includes(item.status)) {
|
||||||
|
item.endTime = '';
|
||||||
|
}
|
||||||
|
return item;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const sanitizeIds = useCallback((item: StructureItem) => {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
const { id, ...rest } = item;
|
||||||
|
return rest;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleUpdateStructures = useCallback(
|
||||||
|
async (newList: StructureItem[]) => {
|
||||||
|
const { added, updated, removed } = getActualStructures(structures, newList);
|
||||||
|
|
||||||
|
const sanitizedAdded = added.map(sanitizeIds);
|
||||||
|
const sanitizedUpdated = updated.map(sanitizeEndTimers);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const { structures: updatedStructures = [] } = await outCommand({
|
||||||
|
type: OutCommand.updateStructures,
|
||||||
|
data: {
|
||||||
|
system_id: systemId,
|
||||||
|
added: sanitizedAdded,
|
||||||
|
updated: sanitizedUpdated,
|
||||||
|
removed,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const finalStructures = updatedStructures.map(mapServerStructure);
|
||||||
|
setStructures(finalStructures);
|
||||||
|
} catch (err) {
|
||||||
|
console.error('Failed to update structures:', err);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[structures, systemId, outCommand, sanitizeIds, sanitizeEndTimers],
|
||||||
|
);
|
||||||
|
|
||||||
|
return { structures, handleUpdateStructures, isLoading, error };
|
||||||
|
}
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SystemStructures';
|
||||||
@@ -0,0 +1,50 @@
|
|||||||
|
// File: TimerCell.tsx
|
||||||
|
import React, { useEffect, useState } from 'react';
|
||||||
|
import { StructureStatus } from '../helpers/structureTypes';
|
||||||
|
import { statusesRequiringTimer } from '../helpers';
|
||||||
|
|
||||||
|
interface TimerCellProps {
|
||||||
|
endTime?: string;
|
||||||
|
status: StructureStatus;
|
||||||
|
}
|
||||||
|
|
||||||
|
function TimerCellImpl({ endTime, status }: TimerCellProps) {
|
||||||
|
const [now, setNow] = useState(() => Date.now());
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!endTime || !statusesRequiringTimer.includes(status)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const intervalId = setInterval(() => {
|
||||||
|
setNow(Date.now());
|
||||||
|
}, 1000);
|
||||||
|
|
||||||
|
return () => clearInterval(intervalId);
|
||||||
|
}, [endTime, status]);
|
||||||
|
|
||||||
|
if (!statusesRequiringTimer.includes(status)) {
|
||||||
|
return <span className="text-stone-400"></span>;
|
||||||
|
}
|
||||||
|
if (!endTime) {
|
||||||
|
return <span className="text-sky-400">Set Timer</span>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const msLeft = new Date(endTime).getTime() - now;
|
||||||
|
if (msLeft <= 0) {
|
||||||
|
return <span className="text-red-500">00:00:00</span>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const sec = Math.floor(msLeft / 1000) % 60;
|
||||||
|
const min = Math.floor(msLeft / (1000 * 60)) % 60;
|
||||||
|
const hr = Math.floor(msLeft / (1000 * 3600));
|
||||||
|
|
||||||
|
const pad = (n: number) => n.toString().padStart(2, '0');
|
||||||
|
return (
|
||||||
|
<span className="text-sky-400">
|
||||||
|
{pad(hr)}:{pad(min)}:{pad(sec)}
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export const TimerCell = React.memo(TimerCellImpl);
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
import { StructureItem } from '../helpers';
|
||||||
|
import { TimerCell } from './TimerCell';
|
||||||
|
|
||||||
|
export function renderTimerCell(row: StructureItem) {
|
||||||
|
return <TimerCell endTime={row.endTime} status={row.status} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function renderOwnerCell(row: StructureItem) {
|
||||||
|
return (
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
{row.ownerId && (
|
||||||
|
<img
|
||||||
|
src={`https://images.evetech.net/corporations/${row.ownerId}/logo?size=32`}
|
||||||
|
alt="corp icon"
|
||||||
|
className="w-5 h-5 object-contain"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<span>{row.ownerTicker || row.ownerName}</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function renderTypeCell(row: StructureItem) {
|
||||||
|
return (
|
||||||
|
<div className="flex items-center gap-1">
|
||||||
|
{row.structureTypeId && (
|
||||||
|
<img
|
||||||
|
src={`https://images.evetech.net/types/${row.structureTypeId}/icon`}
|
||||||
|
alt="icon"
|
||||||
|
className="w-5 h-5 object-contain"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<span>{row.structureType ?? ''}</span>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -2,3 +2,5 @@ export * from './LocalCharacters';
|
|||||||
export * from './SystemInfo';
|
export * from './SystemInfo';
|
||||||
export * from './RoutesWidget';
|
export * from './RoutesWidget';
|
||||||
export * from './SystemSignatures';
|
export * from './SystemSignatures';
|
||||||
|
export * from './SystemStructures';
|
||||||
|
export * from './SystemKills';
|
||||||
|
|||||||
@@ -16,6 +16,8 @@ export const MapRootContent = ({}: MapRootContentProps) => {
|
|||||||
const { interfaceSettings } = useMapRootState();
|
const { interfaceSettings } = useMapRootState();
|
||||||
const { isShowMenu } = interfaceSettings;
|
const { isShowMenu } = interfaceSettings;
|
||||||
|
|
||||||
|
const themeClass = `${interfaceSettings.theme ?? 'default'}-theme`;
|
||||||
|
|
||||||
const [showOnTheMap, setShowOnTheMap] = useState(false);
|
const [showOnTheMap, setShowOnTheMap] = useState(false);
|
||||||
const [showMapSettings, setShowMapSettings] = useState(false);
|
const [showMapSettings, setShowMapSettings] = useState(false);
|
||||||
const mapInterface = <MapInterface />;
|
const mapInterface = <MapInterface />;
|
||||||
@@ -26,27 +28,29 @@ export const MapRootContent = ({}: MapRootContentProps) => {
|
|||||||
useSkipContextMenu();
|
useSkipContextMenu();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Layout map={<MapWrapper />}>
|
<div className={themeClass}>
|
||||||
{!isShowMenu ? (
|
<Layout map={<MapWrapper />}>
|
||||||
<div className="absolute top-0 left-14 w-[calc(100%-3.5rem)] h-[calc(100%-3.5rem)] pointer-events-none">
|
{!isShowMenu ? (
|
||||||
<div className="absolute top-0 left-0 w-[calc(100%-3.5rem)] h-full pointer-events-none">
|
<div className="absolute top-0 left-14 w-[calc(100%-3.5rem)] h-[calc(100%-3.5rem)] pointer-events-none">
|
||||||
<Topbar />
|
<div className="absolute top-0 left-0 w-[calc(100%-3.5rem)] h-full pointer-events-none">
|
||||||
|
<Topbar />
|
||||||
|
{mapInterface}
|
||||||
|
</div>
|
||||||
|
<div className="absolute top-0 right-0 w-14 h-[calc(100%+3.5rem)] pointer-events-auto">
|
||||||
|
<RightBar onShowOnTheMap={handleShowOnTheMap} onShowMapSettings={handleShowMapSettings} />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="absolute top-0 left-14 w-[calc(100%-3.5rem)] h-[calc(100%-3.5rem)] pointer-events-none">
|
||||||
|
<Topbar>
|
||||||
|
<MapContextMenu onShowOnTheMap={handleShowOnTheMap} onShowMapSettings={handleShowMapSettings} />
|
||||||
|
</Topbar>
|
||||||
{mapInterface}
|
{mapInterface}
|
||||||
</div>
|
</div>
|
||||||
<div className="absolute top-0 right-0 w-14 h-[calc(100%+3.5rem)] pointer-events-auto">
|
)}
|
||||||
<RightBar onShowOnTheMap={handleShowOnTheMap} onShowMapSettings={handleShowMapSettings} />
|
<OnTheMap show={showOnTheMap} onHide={() => setShowOnTheMap(false)} />
|
||||||
</div>
|
<MapSettings show={showMapSettings} onHide={() => setShowMapSettings(false)} />
|
||||||
</div>
|
</Layout>
|
||||||
) : (
|
</div>
|
||||||
<div className="absolute top-0 left-14 w-[calc(100%-3.5rem)] h-[calc(100%-3.5rem)] pointer-events-none">
|
|
||||||
<Topbar>
|
|
||||||
<MapContextMenu onShowOnTheMap={handleShowOnTheMap} onShowMapSettings={handleShowMapSettings} />
|
|
||||||
</Topbar>
|
|
||||||
{mapInterface}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<OnTheMap show={showOnTheMap} onHide={() => setShowOnTheMap(false)} />
|
|
||||||
<MapSettings show={showMapSettings} onHide={() => setShowMapSettings(false)} />
|
|
||||||
</Layout>
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -2,7 +2,9 @@ import clsx from 'clsx';
|
|||||||
import classes from './PassageCard.module.scss';
|
import classes from './PassageCard.module.scss';
|
||||||
import { Passage } from '@/hooks/Mapper/types';
|
import { Passage } from '@/hooks/Mapper/types';
|
||||||
import { TimeAgo } from '@/hooks/Mapper/components/ui-kit';
|
import { TimeAgo } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
||||||
import { kgToTons } from '@/hooks/Mapper/utils/kgToTons.ts';
|
import { kgToTons } from '@/hooks/Mapper/utils/kgToTons.ts';
|
||||||
|
import { useMemo } from 'react';
|
||||||
|
|
||||||
type PassageCardType = {
|
type PassageCardType = {
|
||||||
// compact?: boolean;
|
// compact?: boolean;
|
||||||
@@ -26,6 +28,11 @@ export const getShipName = (name: string) => {
|
|||||||
export const PassageCard = ({ inserted_at, character: char, ship }: PassageCardType) => {
|
export const PassageCard = ({ inserted_at, character: char, ship }: PassageCardType) => {
|
||||||
const isOwn = false;
|
const isOwn = false;
|
||||||
|
|
||||||
|
const insertedAt = useMemo(() => {
|
||||||
|
const date = new Date(inserted_at);
|
||||||
|
return date.toLocaleString();
|
||||||
|
}, [inserted_at]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={clsx(classes.CharacterCard, 'w-full text-xs', 'flex flex-col box-border')}>
|
<div className={clsx(classes.CharacterCard, 'w-full text-xs', 'flex flex-col box-border')}>
|
||||||
<div className="flex flex-col justify-between px-2 py-1 gap-1">
|
<div className="flex flex-col justify-between px-2 py-1 gap-1">
|
||||||
@@ -76,7 +83,9 @@ export const PassageCard = ({ inserted_at, character: char, ship }: PassageCardT
|
|||||||
{/*time and class*/}
|
{/*time and class*/}
|
||||||
<div className="flex justify-between">
|
<div className="flex justify-between">
|
||||||
<span className="text-stone-400">
|
<span className="text-stone-400">
|
||||||
<TimeAgo timestamp={inserted_at} />
|
<WdTooltipWrapper content={insertedAt}>
|
||||||
|
<TimeAgo timestamp={inserted_at} />
|
||||||
|
</WdTooltipWrapper>
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<div className="text-stone-400">{kgToTons(parseInt(ship.ship_type_info.mass))}</div>
|
<div className="text-stone-400">{kgToTons(parseInt(ship.ship_type_info.mass))}</div>
|
||||||
|
|||||||
@@ -13,6 +13,7 @@
|
|||||||
.p-tabview-panels {
|
.p-tabview-panels {
|
||||||
padding: 6px 1rem !important;
|
padding: 6px 1rem !important;
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.p-tabview-nav-container {
|
.p-tabview-nav-container {
|
||||||
|
|||||||
@@ -3,8 +3,10 @@ import { Dialog } from 'primereact/dialog';
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
import { TabPanel, TabView } from 'primereact/tabview';
|
import { TabPanel, TabView } from 'primereact/tabview';
|
||||||
import { PrettySwitchbox } from './components';
|
import { PrettySwitchbox } from './components';
|
||||||
import { InterfaceStoredSettings, InterfaceStoredSettingsProps, useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { InterfaceStoredSettingsProps, useMapRootState, InterfaceStoredSettings } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types';
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
import { Dropdown } from 'primereact/dropdown';
|
||||||
|
import { WidgetsSettings } from '@/hooks/Mapper/components/mapRootContent/components/MapSettings/components/WidgetsSettings/WidgetsSettings.tsx';
|
||||||
|
|
||||||
export enum UserSettingsRemoteProps {
|
export enum UserSettingsRemoteProps {
|
||||||
link_signature_on_splash = 'link_signature_on_splash',
|
link_signature_on_splash = 'link_signature_on_splash',
|
||||||
@@ -37,45 +39,101 @@ export interface MapSettingsProps {
|
|||||||
onHide: () => void;
|
onHide: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
type CheckboxesList = {
|
type SettingsListItem = {
|
||||||
prop: keyof UserSettings;
|
prop: keyof UserSettings;
|
||||||
label: string;
|
label: string;
|
||||||
}[];
|
type: 'checkbox' | 'dropdown';
|
||||||
|
options?: { label: string; value: string }[];
|
||||||
|
};
|
||||||
|
|
||||||
const COMMON_CHECKBOXES_PROPS: CheckboxesList = [
|
const COMMON_CHECKBOXES_PROPS: SettingsListItem[] = [
|
||||||
{ prop: InterfaceStoredSettingsProps.isShowMinimap, label: 'Show Minimap' },
|
{
|
||||||
|
prop: InterfaceStoredSettingsProps.isShowMinimap,
|
||||||
|
label: 'Show Minimap',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const SYSTEMS_CHECKBOXES_PROPS: CheckboxesList = [
|
const SYSTEMS_CHECKBOXES_PROPS: SettingsListItem[] = [
|
||||||
{ prop: InterfaceStoredSettingsProps.isShowKSpace, label: 'Highlight Low/High-security systems' },
|
{
|
||||||
{ prop: UserSettingsRemoteProps.select_on_spash, label: 'Auto-select splashed' },
|
prop: InterfaceStoredSettingsProps.isShowKSpace,
|
||||||
|
label: 'Highlight Low/High-security systems',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
prop: UserSettingsRemoteProps.select_on_spash,
|
||||||
|
label: 'Auto-select splashed',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const SIGNATURES_CHECKBOXES_PROPS: CheckboxesList = [
|
const SIGNATURES_CHECKBOXES_PROPS: SettingsListItem[] = [
|
||||||
{ prop: UserSettingsRemoteProps.link_signature_on_splash, label: 'Link signature on splash' },
|
{
|
||||||
{ prop: InterfaceStoredSettingsProps.isShowUnsplashedSignatures, label: 'Show unsplashed signatures' },
|
prop: UserSettingsRemoteProps.link_signature_on_splash,
|
||||||
|
label: 'Link signature on splash',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
prop: InterfaceStoredSettingsProps.isShowUnsplashedSignatures,
|
||||||
|
label: 'Show unsplashed signatures',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const CONNECTIONS_CHECKBOXES_PROPS: CheckboxesList = [
|
const CONNECTIONS_CHECKBOXES_PROPS: SettingsListItem[] = [
|
||||||
{ prop: UserSettingsRemoteProps.delete_connection_with_sigs, label: 'Delete connections to linked signatures' },
|
{
|
||||||
{ prop: InterfaceStoredSettingsProps.isThickConnections, label: 'Thicker connections' },
|
prop: UserSettingsRemoteProps.delete_connection_with_sigs,
|
||||||
|
label: 'Delete connections to linked signatures',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
prop: InterfaceStoredSettingsProps.isThickConnections,
|
||||||
|
label: 'Thicker connections',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
const UI_CHECKBOXES_PROPS: CheckboxesList = [
|
const UI_CHECKBOXES_PROPS: SettingsListItem[] = [
|
||||||
{ prop: InterfaceStoredSettingsProps.isShowMenu, label: 'Enable compact map menu bar' },
|
{
|
||||||
{ prop: InterfaceStoredSettingsProps.isShowBackgroundPattern, label: 'Show background pattern' },
|
prop: InterfaceStoredSettingsProps.isShowMenu,
|
||||||
{ prop: InterfaceStoredSettingsProps.isSoftBackground, label: 'Enable soft background' },
|
label: 'Enable compact map menu bar',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
prop: InterfaceStoredSettingsProps.isShowBackgroundPattern,
|
||||||
|
label: 'Show background pattern',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
prop: InterfaceStoredSettingsProps.isSoftBackground,
|
||||||
|
label: 'Enable soft background',
|
||||||
|
type: 'checkbox',
|
||||||
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
const THEME_OPTIONS = [
|
||||||
|
{ label: 'Default', value: 'default' },
|
||||||
|
{ label: 'Pathfinder', value: 'pathfinder' },
|
||||||
|
];
|
||||||
|
|
||||||
|
const THEME_SETTING: SettingsListItem = {
|
||||||
|
prop: 'theme',
|
||||||
|
label: 'Theme',
|
||||||
|
type: 'dropdown',
|
||||||
|
options: THEME_OPTIONS,
|
||||||
|
};
|
||||||
|
|
||||||
export const MapSettings = ({ show, onHide }: MapSettingsProps) => {
|
export const MapSettings = ({ show, onHide }: MapSettingsProps) => {
|
||||||
const [activeIndex, setActiveIndex] = useState(0);
|
const [activeIndex, setActiveIndex] = useState(0);
|
||||||
const { outCommand, interfaceSettings, setInterfaceSettings } = useMapRootState();
|
const { outCommand, interfaceSettings, setInterfaceSettings } = useMapRootState();
|
||||||
const [userRemoteSettings, setUserRemoteSettings] = useState<UserSettingsRemote>({ ...DEFAULT_REMOTE_SETTINGS });
|
const [userRemoteSettings, setUserRemoteSettings] = useState<UserSettingsRemote>({
|
||||||
|
...DEFAULT_REMOTE_SETTINGS,
|
||||||
|
});
|
||||||
|
|
||||||
const mergedSettings = useMemo(() => {
|
const mergedSettings = useMemo(() => {
|
||||||
return {
|
return {
|
||||||
...interfaceSettings,
|
|
||||||
...userRemoteSettings,
|
...userRemoteSettings,
|
||||||
|
...interfaceSettings,
|
||||||
};
|
};
|
||||||
}, [userRemoteSettings, interfaceSettings]);
|
}, [userRemoteSettings, interfaceSettings]);
|
||||||
|
|
||||||
@@ -84,49 +142,67 @@ export const MapSettings = ({ show, onHide }: MapSettingsProps) => {
|
|||||||
type: OutCommand.getUserSettings,
|
type: OutCommand.getUserSettings,
|
||||||
data: null,
|
data: null,
|
||||||
});
|
});
|
||||||
|
|
||||||
setUserRemoteSettings({
|
setUserRemoteSettings({
|
||||||
...user_settings,
|
...user_settings,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleChangeChecked = useCallback(
|
const handleSettingChange = useCallback(
|
||||||
(prop: keyof UserSettings) => async (checked: boolean) => {
|
async (prop: keyof UserSettings, value: boolean | string) => {
|
||||||
// @ts-ignore
|
if (UserSettingsRemoteList.includes(prop as any)) {
|
||||||
if (UserSettingsRemoteList.includes(prop)) {
|
|
||||||
const newRemoteSettings = {
|
const newRemoteSettings = {
|
||||||
...userRemoteSettings,
|
...userRemoteSettings,
|
||||||
[prop]: checked,
|
[prop]: value,
|
||||||
};
|
};
|
||||||
|
|
||||||
await outCommand({
|
await outCommand({
|
||||||
type: OutCommand.updateUserSettings,
|
type: OutCommand.updateUserSettings,
|
||||||
data: newRemoteSettings,
|
data: newRemoteSettings,
|
||||||
});
|
});
|
||||||
|
|
||||||
setUserRemoteSettings(newRemoteSettings);
|
setUserRemoteSettings(newRemoteSettings);
|
||||||
return;
|
} else {
|
||||||
|
setInterfaceSettings({
|
||||||
|
...interfaceSettings,
|
||||||
|
[prop]: value,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
setInterfaceSettings({
|
|
||||||
...interfaceSettings,
|
|
||||||
[prop]: checked,
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
[interfaceSettings, outCommand, setInterfaceSettings, userRemoteSettings],
|
[userRemoteSettings, interfaceSettings, outCommand, setInterfaceSettings],
|
||||||
);
|
);
|
||||||
|
|
||||||
const renderCheckboxesList = (list: CheckboxesList) => {
|
const renderSettingItem = (item: SettingsListItem) => {
|
||||||
return list.map(x => {
|
const currentValue = mergedSettings[item.prop];
|
||||||
|
|
||||||
|
if (item.type === 'checkbox') {
|
||||||
return (
|
return (
|
||||||
<PrettySwitchbox
|
<PrettySwitchbox
|
||||||
key={x.prop}
|
key={item.prop}
|
||||||
label={x.label}
|
label={item.label}
|
||||||
checked={mergedSettings[x.prop]}
|
checked={!!currentValue}
|
||||||
setChecked={handleChangeChecked(x.prop)}
|
setChecked={checked => handleSettingChange(item.prop, checked)}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
}
|
||||||
|
|
||||||
|
if (item.type === 'dropdown' && item.options) {
|
||||||
|
return (
|
||||||
|
<div key={item.prop} className="flex items-center gap-2 mt-2">
|
||||||
|
<label className="text-sm">{item.label}:</label>
|
||||||
|
<Dropdown
|
||||||
|
className="text-sm"
|
||||||
|
value={currentValue}
|
||||||
|
options={item.options}
|
||||||
|
onChange={e => handleSettingChange(item.prop, e.value)}
|
||||||
|
placeholder="Select a theme"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderSettingsList = (list: SettingsListItem[]) => {
|
||||||
|
return list.map(renderSettingItem);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -137,10 +213,7 @@ export const MapSettings = ({ show, onHide }: MapSettingsProps) => {
|
|||||||
style={{ width: '550px' }}
|
style={{ width: '550px' }}
|
||||||
onShow={handleShow}
|
onShow={handleShow}
|
||||||
onHide={() => {
|
onHide={() => {
|
||||||
if (!show) {
|
if (!show) return;
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setActiveIndex(0);
|
setActiveIndex(0);
|
||||||
onHide();
|
onHide();
|
||||||
}}
|
}}
|
||||||
@@ -148,27 +221,33 @@ export const MapSettings = ({ show, onHide }: MapSettingsProps) => {
|
|||||||
<div className="flex flex-col gap-3">
|
<div className="flex flex-col gap-3">
|
||||||
<div className="flex flex-col gap-2">
|
<div className="flex flex-col gap-2">
|
||||||
<div className={styles.verticalTabsContainer}>
|
<div className={styles.verticalTabsContainer}>
|
||||||
<TabView
|
<TabView activeIndex={activeIndex} onTabChange={e => setActiveIndex(e.index)}>
|
||||||
activeIndex={activeIndex}
|
|
||||||
onTabChange={e => setActiveIndex(e.index)}
|
|
||||||
className={styles.verticalTabView}
|
|
||||||
>
|
|
||||||
<TabPanel header="Common" headerClassName={styles.verticalTabHeader}>
|
<TabPanel header="Common" headerClassName={styles.verticalTabHeader}>
|
||||||
<div className="w-full h-full flex flex-col gap-1">{renderCheckboxesList(COMMON_CHECKBOXES_PROPS)}</div>
|
<div className="w-full h-full flex flex-col gap-1">{renderSettingsList(COMMON_CHECKBOXES_PROPS)}</div>
|
||||||
</TabPanel>
|
</TabPanel>
|
||||||
|
|
||||||
<TabPanel header="Systems" headerClassName={styles.verticalTabHeader}>
|
<TabPanel header="Systems" headerClassName={styles.verticalTabHeader}>
|
||||||
<div className="w-full h-full flex flex-col gap-1">
|
<div className="w-full h-full flex flex-col gap-1">{renderSettingsList(SYSTEMS_CHECKBOXES_PROPS)}</div>
|
||||||
{renderCheckboxesList(SYSTEMS_CHECKBOXES_PROPS)}
|
|
||||||
</div>
|
|
||||||
</TabPanel>
|
</TabPanel>
|
||||||
|
|
||||||
<TabPanel header="Connections" headerClassName={styles.verticalTabHeader}>
|
<TabPanel header="Connections" headerClassName={styles.verticalTabHeader}>
|
||||||
{renderCheckboxesList(CONNECTIONS_CHECKBOXES_PROPS)}
|
{renderSettingsList(CONNECTIONS_CHECKBOXES_PROPS)}
|
||||||
</TabPanel>
|
</TabPanel>
|
||||||
|
|
||||||
<TabPanel header="Signatures" headerClassName={styles.verticalTabHeader}>
|
<TabPanel header="Signatures" headerClassName={styles.verticalTabHeader}>
|
||||||
{renderCheckboxesList(SIGNATURES_CHECKBOXES_PROPS)}
|
{renderSettingsList(SIGNATURES_CHECKBOXES_PROPS)}
|
||||||
</TabPanel>
|
</TabPanel>
|
||||||
|
|
||||||
<TabPanel header="User Interface" headerClassName={styles.verticalTabHeader}>
|
<TabPanel header="User Interface" headerClassName={styles.verticalTabHeader}>
|
||||||
{renderCheckboxesList(UI_CHECKBOXES_PROPS)}
|
{renderSettingsList(UI_CHECKBOXES_PROPS)}
|
||||||
|
</TabPanel>
|
||||||
|
|
||||||
|
<TabPanel header="Widgets" className="h-full" headerClassName={styles.verticalTabHeader}>
|
||||||
|
<WidgetsSettings />
|
||||||
|
</TabPanel>
|
||||||
|
|
||||||
|
<TabPanel header="Theme" headerClassName={styles.verticalTabHeader}>
|
||||||
|
{renderSettingItem(THEME_SETTING)}
|
||||||
</TabPanel>
|
</TabPanel>
|
||||||
</TabView>
|
</TabView>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import styles from './MapSettings.module.scss';
|
import styles from './MapSettings.module.scss';
|
||||||
|
|
||||||
import { WdCheckbox } from '@/hooks/Mapper/components/ui-kit';
|
import { WdCheckbox } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
|
||||||
interface PrettySwitchboxProps {
|
interface PrettySwitchboxProps {
|
||||||
|
|||||||
@@ -0,0 +1,40 @@
|
|||||||
|
import { PrettySwitchbox } from '@/hooks/Mapper/components/mapRootContent/components/MapSettings/components';
|
||||||
|
import { getWidgetsCheckboxesProps, WidgetsIds } from '@/hooks/Mapper/components/mapInterface/constants.tsx';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useCallback } from 'react';
|
||||||
|
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
|
||||||
|
export interface WidgetsSettingsProps {}
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-empty-pattern
|
||||||
|
export const WidgetsSettings = ({}: WidgetsSettingsProps) => {
|
||||||
|
const { windowsSettings, toggleWidgetVisibility, resetWidgets, data } = useMapRootState();
|
||||||
|
|
||||||
|
const handleWidgetSettingsChange = useCallback(
|
||||||
|
(widget: WidgetsIds) => toggleWidgetVisibility(widget),
|
||||||
|
[toggleWidgetVisibility],
|
||||||
|
);
|
||||||
|
|
||||||
|
const detailedKillsDisabled = data.options?.detailedKillsDisabled === true;
|
||||||
|
const widgetProps = getWidgetsCheckboxesProps(detailedKillsDisabled);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex flex-col h-full gap-2">
|
||||||
|
<div>
|
||||||
|
{widgetProps.map(widget => (
|
||||||
|
<PrettySwitchbox
|
||||||
|
key={widget.id}
|
||||||
|
label={widget.label}
|
||||||
|
checked={windowsSettings.visible.some(x => x === widget.id)}
|
||||||
|
setChecked={() => handleWidgetSettingsChange(widget.id)}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
<div className="grid grid-cols-[1fr_auto]">
|
||||||
|
<div />
|
||||||
|
<Button className="py-[4px]" onClick={resetWidgets} outlined size="small" label="Reset Widgets"></Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -53,6 +53,7 @@ export const SignatureSettings = ({ systemId, show, onHide, signatureData }: Map
|
|||||||
...out,
|
...out,
|
||||||
custom_info: JSON.stringify({
|
custom_info: JSON.stringify({
|
||||||
k162Type: values.k162Type,
|
k162Type: values.k162Type,
|
||||||
|
isEOL: values.isEOL,
|
||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -127,14 +128,17 @@ export const SignatureSettings = ({ systemId, show, onHide, signatureData }: Map
|
|||||||
const { linked_system, custom_info, ...rest } = signatureData;
|
const { linked_system, custom_info, ...rest } = signatureData;
|
||||||
|
|
||||||
let k162Type = null;
|
let k162Type = null;
|
||||||
|
let isEOL = false;
|
||||||
if (custom_info) {
|
if (custom_info) {
|
||||||
const customInfo = JSON.parse(custom_info);
|
const customInfo = JSON.parse(custom_info);
|
||||||
k162Type = customInfo.k162Type;
|
k162Type = customInfo.k162Type;
|
||||||
|
isEOL = customInfo.isEOL;
|
||||||
}
|
}
|
||||||
|
|
||||||
signatureForm.reset({
|
signatureForm.reset({
|
||||||
linked_system: linked_system?.solar_system_id.toString() ?? undefined,
|
linked_system: linked_system?.solar_system_id.toString() ?? undefined,
|
||||||
k162Type: k162Type,
|
k162Type: k162Type,
|
||||||
|
isEOL: isEOL,
|
||||||
...rest,
|
...rest,
|
||||||
});
|
});
|
||||||
}, [signatureForm, signatureData]);
|
}, [signatureForm, signatureData]);
|
||||||
|
|||||||
@@ -0,0 +1,24 @@
|
|||||||
|
import { InputSwitch } from 'primereact/inputswitch';
|
||||||
|
import { Controller, useFormContext } from 'react-hook-form';
|
||||||
|
import { SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
|
export interface SignatureEOLCheckboxProps {
|
||||||
|
name: string;
|
||||||
|
defaultValue?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SignatureEOLCheckbox = ({ name, defaultValue = false }: SignatureEOLCheckboxProps) => {
|
||||||
|
const { control } = useFormContext<SystemSignature>();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Controller
|
||||||
|
// @ts-ignore
|
||||||
|
name={name}
|
||||||
|
control={control}
|
||||||
|
defaultValue={defaultValue}
|
||||||
|
render={({ field }) => {
|
||||||
|
return <InputSwitch className="my-1" checked={!!field.value} onChange={e => field.onChange(e.value)} />;
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './SignatureEOLCheckbox.tsx';
|
||||||
@@ -3,6 +3,7 @@ import { SystemSignature } from '@/hooks/Mapper/types';
|
|||||||
import { SignatureWormholeTypeSelect } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureWormholeTypeSelect';
|
import { SignatureWormholeTypeSelect } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureWormholeTypeSelect';
|
||||||
import { SignatureK162TypeSelect } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureK162TypeSelect';
|
import { SignatureK162TypeSelect } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureK162TypeSelect';
|
||||||
import { SignatureLeadsToSelect } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureLeadsToSelect';
|
import { SignatureLeadsToSelect } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureLeadsToSelect';
|
||||||
|
import { SignatureEOLCheckbox } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings/components/SignatureEOLCheckbox';
|
||||||
|
|
||||||
export const SignatureGroupContentWormholes = () => {
|
export const SignatureGroupContentWormholes = () => {
|
||||||
const { watch } = useFormContext<SystemSignature>();
|
const { watch } = useFormContext<SystemSignature>();
|
||||||
@@ -26,6 +27,11 @@ export const SignatureGroupContentWormholes = () => {
|
|||||||
<span>Leads To:</span>
|
<span>Leads To:</span>
|
||||||
<SignatureLeadsToSelect name="linked_system" />
|
<SignatureLeadsToSelect name="linked_system" />
|
||||||
</label>
|
</label>
|
||||||
|
|
||||||
|
<label className="grid grid-cols-[100px_250px_1fr] gap-2 items-center text-[14px]">
|
||||||
|
<span>EOL:</span>
|
||||||
|
<SignatureEOLCheckbox name="isEOL" />
|
||||||
|
</label>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -3,100 +3,8 @@ import clsx from 'clsx';
|
|||||||
import { Controller, useFormContext } from 'react-hook-form';
|
import { Controller, useFormContext } from 'react-hook-form';
|
||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { SystemSignature } from '@/hooks/Mapper/types';
|
import { SystemSignature } from '@/hooks/Mapper/types';
|
||||||
import { WHClassView } from '@/hooks/Mapper/components/ui-kit';
|
import { K162_TYPES } from '@/hooks/Mapper/constants.ts';
|
||||||
|
import { renderK162Type } from '.';
|
||||||
export const k162Types = [
|
|
||||||
{
|
|
||||||
label: 'Hi-Sec',
|
|
||||||
value: 'hs',
|
|
||||||
whClassName: 'A641',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Low-Sec',
|
|
||||||
value: 'ls',
|
|
||||||
whClassName: 'J377',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Null-Sec',
|
|
||||||
value: 'ns',
|
|
||||||
whClassName: 'C248',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'C1',
|
|
||||||
value: 'c1',
|
|
||||||
whClassName: 'E004',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'C2',
|
|
||||||
value: 'c2',
|
|
||||||
whClassName: 'D382',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'C3',
|
|
||||||
value: 'c3',
|
|
||||||
whClassName: 'L477',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'C4',
|
|
||||||
value: 'c4',
|
|
||||||
whClassName: 'M001',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'C5',
|
|
||||||
value: 'c5',
|
|
||||||
whClassName: 'L614',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'C6',
|
|
||||||
value: 'c6',
|
|
||||||
whClassName: 'G008',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'C13',
|
|
||||||
value: 'c13',
|
|
||||||
whClassName: 'A009',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Thera',
|
|
||||||
value: 'thera',
|
|
||||||
whClassName: 'F353',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: 'Pochven',
|
|
||||||
value: 'pochven',
|
|
||||||
whClassName: 'F216',
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
const renderNoValue = () => <div className="flex gap-2 items-center">-Unknown-</div>;
|
|
||||||
|
|
||||||
// @ts-ignore
|
|
||||||
export const renderK162Type = (option: {
|
|
||||||
label?: string;
|
|
||||||
value: string;
|
|
||||||
security?: string;
|
|
||||||
system_class?: number;
|
|
||||||
whClassName?: string;
|
|
||||||
}) => {
|
|
||||||
if (!option) {
|
|
||||||
return renderNoValue();
|
|
||||||
}
|
|
||||||
const { value, whClassName = '' } = option;
|
|
||||||
if (value == null) {
|
|
||||||
return renderNoValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<WHClassView
|
|
||||||
classNameWh="!text-[11px] !font-bold"
|
|
||||||
hideWhClassName
|
|
||||||
hideTooltip
|
|
||||||
whClassName={whClassName}
|
|
||||||
noOffset
|
|
||||||
useShortTitle
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export interface SignatureK162TypeSelectProps {
|
export interface SignatureK162TypeSelectProps {
|
||||||
name: string;
|
name: string;
|
||||||
@@ -107,7 +15,7 @@ export const SignatureK162TypeSelect = ({ name, defaultValue = '' }: SignatureK1
|
|||||||
const { control } = useFormContext<SystemSignature>();
|
const { control } = useFormContext<SystemSignature>();
|
||||||
|
|
||||||
const options = useMemo(() => {
|
const options = useMemo(() => {
|
||||||
return [{ value: null }, ...k162Types];
|
return [{ value: null }, ...K162_TYPES];
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -1 +1,2 @@
|
|||||||
export * from './SignatureK162TypeSelect.tsx';
|
export * from './SignatureK162TypeSelect.tsx';
|
||||||
|
export * from './renderK162Type.tsx';
|
||||||
|
|||||||
@@ -0,0 +1,26 @@
|
|||||||
|
import { WHClassView } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { K162Type } from '@/hooks/Mapper/constants.ts';
|
||||||
|
|
||||||
|
const renderNoValue = () => <div className="flex gap-2 items-center">-Unknown-</div>;
|
||||||
|
|
||||||
|
export const renderK162Type = (option: K162Type) => {
|
||||||
|
if (!option) {
|
||||||
|
return renderNoValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
const { value, whClassName = '' } = option;
|
||||||
|
if (value == null) {
|
||||||
|
return renderNoValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<WHClassView
|
||||||
|
classNameWh="!text-[11px] !font-bold"
|
||||||
|
hideWhClassName
|
||||||
|
hideTooltip
|
||||||
|
whClassName={whClassName}
|
||||||
|
noOffset
|
||||||
|
useShortTitle
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -14,9 +14,10 @@ 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 { Commands } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { Node, XYPosition } from 'reactflow';
|
import { Node, XYPosition } from 'reactflow';
|
||||||
|
|
||||||
import { Commands } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { useCommandsSystems } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
import { emitMapEvent, useMapEventListener } from '@/hooks/Mapper/events';
|
import { emitMapEvent, 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';
|
||||||
@@ -32,7 +33,7 @@ export const MapWrapper = () => {
|
|||||||
const {
|
const {
|
||||||
update,
|
update,
|
||||||
outCommand,
|
outCommand,
|
||||||
data: { selectedConnections, selectedSystems, hubs, systems },
|
data: { selectedConnections, selectedSystems, hubs, systems, connections, linkSignatureToSystem },
|
||||||
interfaceSettings: {
|
interfaceSettings: {
|
||||||
isShowMenu,
|
isShowMenu,
|
||||||
isShowMinimap = STORED_INTERFACE_DEFAULT_VALUES.isShowMinimap,
|
isShowMinimap = STORED_INTERFACE_DEFAULT_VALUES.isShowMinimap,
|
||||||
@@ -40,30 +41,25 @@ export const MapWrapper = () => {
|
|||||||
isThickConnections,
|
isThickConnections,
|
||||||
isShowBackgroundPattern,
|
isShowBackgroundPattern,
|
||||||
isSoftBackground,
|
isSoftBackground,
|
||||||
|
theme,
|
||||||
},
|
},
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
const { deleteSystems } = useDeleteSystems();
|
const { deleteSystems } = useDeleteSystems();
|
||||||
const { mapRef, runCommand } = useCommonMapEventProcessor();
|
const { mapRef, runCommand } = useCommonMapEventProcessor();
|
||||||
|
|
||||||
|
const { updateLinkSignatureToSystem } = useCommandsSystems();
|
||||||
const { open, ...systemContextProps } = useContextMenuSystemHandlers({ systems, hubs, outCommand });
|
const { open, ...systemContextProps } = useContextMenuSystemHandlers({ systems, hubs, outCommand });
|
||||||
const { handleSystemMultipleContext, ...systemMultipleCtxProps } = useContextMenuSystemMultipleHandlers();
|
const { handleSystemMultipleContext, ...systemMultipleCtxProps } = useContextMenuSystemMultipleHandlers();
|
||||||
|
|
||||||
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 [openAddSystem, setOpenAddSystem] = useState<XYPosition | null>(null);
|
const [openAddSystem, setOpenAddSystem] = useState<XYPosition | null>(null);
|
||||||
const [selectedConnection, setSelectedConnection] = useState<SolarSystemConnection | null>(null);
|
const [selectedConnection, setSelectedConnection] = useState<SolarSystemConnection | null>(null);
|
||||||
|
|
||||||
const ref = useRef({ selectedConnections, selectedSystems, systemContextProps, systems, deleteSystems });
|
const ref = useRef({ selectedConnections, selectedSystems, systemContextProps, systems, connections, deleteSystems });
|
||||||
ref.current = { selectedConnections, selectedSystems, systemContextProps, systems, deleteSystems };
|
ref.current = { selectedConnections, selectedSystems, systemContextProps, systems, connections, deleteSystems };
|
||||||
|
|
||||||
useMapEventListener(event => {
|
useMapEventListener(event => {
|
||||||
switch (event.name) {
|
|
||||||
case Commands.linkSignatureToSystem:
|
|
||||||
setOpenLinkSignatures(event.data);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
runCommand(event);
|
runCommand(event);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -92,9 +88,6 @@ export const MapWrapper = () => {
|
|||||||
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);
|
||||||
}
|
}
|
||||||
@@ -132,6 +125,11 @@ export const MapWrapper = () => {
|
|||||||
setOpenAddSystem(coordinates);
|
setOpenAddSystem(coordinates);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const canRemoveConnection = useCallback((connectionId: string) => {
|
||||||
|
const { connections } = ref.current;
|
||||||
|
return !connections.some(x => x.id === connectionId);
|
||||||
|
}, []);
|
||||||
|
|
||||||
const handleSubmitAddSystem: SearchOnSubmitCallback = useCallback(
|
const handleSubmitAddSystem: SearchOnSubmitCallback = useCallback(
|
||||||
async item => {
|
async item => {
|
||||||
if (ref.current.systems.some(x => x.system_static_info.solar_system_id === item.value)) {
|
if (ref.current.systems.some(x => x.system_static_info.solar_system_id === item.value)) {
|
||||||
@@ -166,7 +164,9 @@ export const MapWrapper = () => {
|
|||||||
isThickConnections={isThickConnections}
|
isThickConnections={isThickConnections}
|
||||||
isShowBackgroundPattern={isShowBackgroundPattern}
|
isShowBackgroundPattern={isShowBackgroundPattern}
|
||||||
isSoftBackground={isSoftBackground}
|
isSoftBackground={isSoftBackground}
|
||||||
|
theme={theme}
|
||||||
onAddSystem={onAddSystem}
|
onAddSystem={onAddSystem}
|
||||||
|
canRemoveConnection={canRemoveConnection}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{openSettings != null && (
|
{openSettings != null && (
|
||||||
@@ -177,8 +177,8 @@ export const MapWrapper = () => {
|
|||||||
<SystemCustomLabelDialog systemId={openCustomLabel} visible setVisible={() => setOpenCustomLabel(null)} />
|
<SystemCustomLabelDialog systemId={openCustomLabel} visible setVisible={() => setOpenCustomLabel(null)} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{openLinkSignatures != null && (
|
{linkSignatureToSystem != null && (
|
||||||
<SystemLinkSignatureDialog data={openLinkSignatures} setVisible={() => setOpenLinkSignatures(null)} />
|
<SystemLinkSignatureDialog data={linkSignatureToSystem} setVisible={() => updateLinkSignatureToSystem(null)} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<AddSystemDialog
|
<AddSystemDialog
|
||||||
|
|||||||
@@ -15,7 +15,7 @@
|
|||||||
border-style: solid;
|
border-style: solid;
|
||||||
border-color: #272727;
|
border-color: #272727;
|
||||||
background-color: rgba(0, 0, 0, 0);
|
background-color: rgba(0, 0, 0, 0);
|
||||||
border-radius: 3px;
|
border-radius: 0 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.CharName {
|
.CharName {
|
||||||
@@ -26,7 +26,9 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.CharIcon {}
|
.CharIcon {
|
||||||
|
border-radius: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
.CharRow {
|
.CharRow {
|
||||||
display: grid;
|
display: grid;
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import clsx from 'clsx';
|
|||||||
import classes from './CharacterCard.module.scss';
|
import classes from './CharacterCard.module.scss';
|
||||||
import { SystemView } from '@/hooks/Mapper/components/ui-kit/SystemView';
|
import { SystemView } from '@/hooks/Mapper/components/ui-kit/SystemView';
|
||||||
import { CharacterTypeRaw, WithIsOwnCharacter } from '@/hooks/Mapper/types';
|
import { CharacterTypeRaw, WithIsOwnCharacter } from '@/hooks/Mapper/types';
|
||||||
import { Commands } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { Commands } from '@/hooks/Mapper/types/mapHandlers';
|
||||||
import { emitMapEvent } from '@/hooks/Mapper/events';
|
import { emitMapEvent } from '@/hooks/Mapper/events';
|
||||||
|
|
||||||
type CharacterCardProps = {
|
type CharacterCardProps = {
|
||||||
@@ -18,12 +18,8 @@ const SHIP_NAME_RX = /u'|'/g;
|
|||||||
export const getShipName = (name: string) => {
|
export const getShipName = (name: string) => {
|
||||||
return name
|
return name
|
||||||
.replace(SHIP_NAME_RX, '')
|
.replace(SHIP_NAME_RX, '')
|
||||||
.replace(/\\u([\dA-Fa-f]{4})/g, (_, grp) => {
|
.replace(/\\u([\dA-Fa-f]{4})/g, (_, grp) => String.fromCharCode(parseInt(grp, 16)))
|
||||||
return String.fromCharCode(parseInt(grp, 16));
|
.replace(/\\x([\dA-Fa-f]{2})/g, (_, grp) => String.fromCharCode(parseInt(grp, 16)));
|
||||||
})
|
|
||||||
.replace(/\\x([\dA-Fa-f]{2})/g, (_, grp) => {
|
|
||||||
return String.fromCharCode(parseInt(grp, 16));
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const CharacterCard = ({
|
export const CharacterCard = ({
|
||||||
@@ -47,7 +43,9 @@ export const CharacterCard = ({
|
|||||||
{!compact && (
|
{!compact && (
|
||||||
<span
|
<span
|
||||||
className={clsx(classes.EveIcon, classes.CharIcon, 'wd-bg-default')}
|
className={clsx(classes.EveIcon, classes.CharIcon, 'wd-bg-default')}
|
||||||
style={{ backgroundImage: `url(https://images.evetech.net/characters/${char.eve_id}/portrait)` }}
|
style={{
|
||||||
|
backgroundImage: `url(https://images.evetech.net/characters/${char.eve_id}/portrait)`,
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<div className="flex flex-col flex-grow">
|
<div className="flex flex-col flex-grow">
|
||||||
|
|||||||
@@ -1,20 +1,8 @@
|
|||||||
import React, {
|
import React, { ForwardedRef, forwardRef, useCallback, useEffect, useImperativeHandle, useRef, useState } from 'react';
|
||||||
ForwardedRef,
|
|
||||||
forwardRef,
|
|
||||||
MouseEvent,
|
|
||||||
MouseEventHandler,
|
|
||||||
useCallback,
|
|
||||||
useEffect,
|
|
||||||
useImperativeHandle,
|
|
||||||
useRef,
|
|
||||||
useState,
|
|
||||||
} from 'react';
|
|
||||||
import { createPortal } from 'react-dom';
|
import { createPortal } from 'react-dom';
|
||||||
|
|
||||||
import classes from './WdTooltip.module.scss';
|
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import debounce from 'lodash.debounce';
|
import debounce from 'lodash.debounce';
|
||||||
import { WithClassName } from '@/hooks/Mapper/types/common.ts';
|
import classes from './WdTooltip.module.scss';
|
||||||
|
|
||||||
export enum TooltipPosition {
|
export enum TooltipPosition {
|
||||||
default = 'default',
|
default = 'default',
|
||||||
@@ -29,11 +17,7 @@ export interface TooltipProps {
|
|||||||
offset?: number;
|
offset?: number;
|
||||||
content: (() => React.ReactNode) | React.ReactNode;
|
content: (() => React.ReactNode) | React.ReactNode;
|
||||||
targetSelector?: string;
|
targetSelector?: string;
|
||||||
}
|
interactive?: boolean;
|
||||||
|
|
||||||
export interface WdTooltipHandlers {
|
|
||||||
show: MouseEventHandler;
|
|
||||||
hide: MouseEventHandler;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface OffsetPosition {
|
export interface OffsetPosition {
|
||||||
@@ -41,169 +25,185 @@ export interface OffsetPosition {
|
|||||||
left: number;
|
left: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
// eslint-disable-next-line react/display-name
|
export interface WdTooltipHandlers {
|
||||||
export const WdTooltip = forwardRef((props: TooltipProps & WithClassName, ref: ForwardedRef<WdTooltipHandlers>) => {
|
show: (e?: React.MouseEvent) => void;
|
||||||
const { content, targetSelector, position: tPosition = TooltipPosition.default, className, offset = 5 } = props;
|
hide: (e?: React.MouseEvent) => void;
|
||||||
|
getIsMouseInside: () => boolean;
|
||||||
|
}
|
||||||
|
|
||||||
const [visible, setVisible] = useState(false);
|
export const WdTooltip = forwardRef(
|
||||||
const [position, setPosition] = useState<OffsetPosition | null>(null);
|
(props: TooltipProps & { className?: string }, ref: ForwardedRef<WdTooltipHandlers>) => {
|
||||||
const [ev, setEv] = useState<MouseEvent>();
|
const {
|
||||||
const tooltipRef = useRef<HTMLDivElement>(null);
|
content,
|
||||||
|
targetSelector,
|
||||||
|
position: tPosition = TooltipPosition.default,
|
||||||
|
className,
|
||||||
|
offset = 5,
|
||||||
|
interactive = false,
|
||||||
|
} = props;
|
||||||
|
|
||||||
const calcTooltipPosition = useCallback(({ x, y }: { x: number; y: number }) => {
|
const [visible, setVisible] = useState(false);
|
||||||
let newLeft = x;
|
const [pos, setPos] = useState<OffsetPosition | null>(null);
|
||||||
let newTop = y;
|
const [ev, setEv] = useState<React.MouseEvent>();
|
||||||
|
const tooltipRef = useRef<HTMLDivElement>(null);
|
||||||
|
const [isMouseInsideTooltip, setIsMouseInsideTooltip] = useState(false);
|
||||||
|
|
||||||
if (!tooltipRef.current) {
|
const calcTooltipPosition = useCallback(({ x, y }: { x: number; y: number }) => {
|
||||||
|
if (!tooltipRef.current) return { left: x, top: y };
|
||||||
|
const tooltipWidth = tooltipRef.current.offsetWidth;
|
||||||
|
const tooltipHeight = tooltipRef.current.offsetHeight;
|
||||||
|
let newLeft = x;
|
||||||
|
let newTop = y;
|
||||||
|
|
||||||
|
if (newLeft < 0) newLeft = 10;
|
||||||
|
if (newTop < 0) newTop = 10;
|
||||||
|
if (newLeft + tooltipWidth + 10 > window.innerWidth) {
|
||||||
|
newLeft = window.innerWidth - tooltipWidth - 10;
|
||||||
|
}
|
||||||
|
if (newTop + tooltipHeight + 10 > window.innerHeight) {
|
||||||
|
newTop = window.innerHeight - tooltipHeight - 10;
|
||||||
|
}
|
||||||
return { left: newLeft, top: newTop };
|
return { left: newLeft, top: newTop };
|
||||||
}
|
}, []);
|
||||||
|
|
||||||
const tooltipWidth = tooltipRef.current.offsetWidth;
|
useImperativeHandle(ref, () => ({
|
||||||
const tooltipHeight = tooltipRef.current.offsetHeight;
|
show: (mouseEvt?: React.MouseEvent) => {
|
||||||
|
if (mouseEvt) setEv(mouseEvt);
|
||||||
|
setPos(null);
|
||||||
|
setVisible(true);
|
||||||
|
},
|
||||||
|
hide: () => {
|
||||||
|
setVisible(false);
|
||||||
|
},
|
||||||
|
getIsMouseInside: () => isMouseInsideTooltip,
|
||||||
|
}));
|
||||||
|
|
||||||
if (newLeft < 0) {
|
useEffect(() => {
|
||||||
newLeft = 10;
|
if (!tooltipRef.current || !ev) return;
|
||||||
}
|
const tooltipEl = tooltipRef.current;
|
||||||
|
const { clientX, clientY, target } = ev;
|
||||||
|
const targetBounds = (target as HTMLElement).getBoundingClientRect();
|
||||||
|
|
||||||
if (newTop < 0) {
|
let offsetX = clientX;
|
||||||
newTop = 10;
|
let offsetY = clientY;
|
||||||
}
|
|
||||||
|
|
||||||
if (newLeft + tooltipWidth + 10 > window.innerWidth) {
|
if (tPosition === TooltipPosition.left) {
|
||||||
newLeft = window.innerWidth - tooltipWidth - 10;
|
const tooltipBounds = tooltipEl.getBoundingClientRect();
|
||||||
}
|
offsetX = targetBounds.left - tooltipBounds.width - offset;
|
||||||
if (newTop + tooltipHeight + 10 > window.innerHeight) {
|
offsetY = targetBounds.y + targetBounds.height / 2 - tooltipBounds.height / 2;
|
||||||
newTop = window.innerHeight - tooltipHeight - 10;
|
if (offsetX <= 0) {
|
||||||
}
|
offsetX = targetBounds.left + targetBounds.width + offset;
|
||||||
return { left: newLeft, top: newTop };
|
}
|
||||||
}, []);
|
setPos(calcTooltipPosition({ x: offsetX, y: offsetY }));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
if (tPosition === TooltipPosition.right) {
|
||||||
if (!tooltipRef.current || !ev) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { clientX, clientY, target } = ev;
|
|
||||||
|
|
||||||
const targetBounds = (target as HTMLElement).getBoundingClientRect();
|
|
||||||
const tooltipBounds = tooltipRef.current.getBoundingClientRect();
|
|
||||||
|
|
||||||
let offsetX = clientX;
|
|
||||||
let offsetY = clientY;
|
|
||||||
|
|
||||||
if (tPosition === TooltipPosition.left) {
|
|
||||||
offsetX = targetBounds.left - tooltipBounds.width - offset;
|
|
||||||
offsetY = targetBounds.y + targetBounds.height / 2 - tooltipBounds.height / 2;
|
|
||||||
|
|
||||||
if (offsetX <= 0) {
|
|
||||||
offsetX = targetBounds.left + targetBounds.width + offset;
|
offsetX = targetBounds.left + targetBounds.width + offset;
|
||||||
}
|
offsetY = targetBounds.y + targetBounds.height / 2 - tooltipEl.offsetHeight / 2;
|
||||||
|
setPos(calcTooltipPosition({ x: offsetX, y: offsetY }));
|
||||||
setPosition(calcTooltipPosition({ x: offsetX, y: offsetY }));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tPosition === TooltipPosition.right) {
|
|
||||||
offsetX = targetBounds.left + targetBounds.width + offset;
|
|
||||||
offsetY = targetBounds.y + targetBounds.height / 2 - tooltipBounds.height / 2;
|
|
||||||
|
|
||||||
setPosition(calcTooltipPosition({ x: offsetX, y: offsetY }));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tPosition === TooltipPosition.top) {
|
|
||||||
offsetY = targetBounds.top - tooltipBounds.height - offset;
|
|
||||||
offsetX = targetBounds.x + targetBounds.width / 2 - tooltipBounds.width / 2;
|
|
||||||
|
|
||||||
setPosition(calcTooltipPosition({ x: offsetX, y: offsetY }));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// default case
|
|
||||||
setPosition(calcTooltipPosition({ x: clientX, y: clientY }));
|
|
||||||
}, [calcTooltipPosition, ev, tPosition, offset]);
|
|
||||||
|
|
||||||
useImperativeHandle(ref, () => ({
|
|
||||||
show: e => {
|
|
||||||
setEv(e);
|
|
||||||
setVisible(true);
|
|
||||||
setPosition(null);
|
|
||||||
},
|
|
||||||
hide: () => {
|
|
||||||
setVisible(false);
|
|
||||||
},
|
|
||||||
}));
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (targetSelector == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const handleMouseMove = (e: MouseEvent) => {
|
|
||||||
const targetElement = e.target as HTMLElement;
|
|
||||||
|
|
||||||
if (!targetElement) {
|
|
||||||
setVisible(false);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const nodesFound = [...(targetElement?.parentElement?.querySelectorAll(targetSelector) ?? [])];
|
if (tPosition === TooltipPosition.top) {
|
||||||
|
offsetY = targetBounds.top - tooltipEl.offsetHeight - offset;
|
||||||
if (!nodesFound.includes(targetElement)) {
|
offsetX = targetBounds.x + targetBounds.width / 2 - tooltipEl.offsetWidth / 2;
|
||||||
setVisible(false);
|
setPos(calcTooltipPosition({ x: offsetX, y: offsetY }));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
setVisible(true);
|
if (tPosition === TooltipPosition.bottom) {
|
||||||
if (tooltipRef.current) {
|
offsetY = targetBounds.bottom + offset;
|
||||||
const { clientX, clientY } = e;
|
offsetX = targetBounds.x + targetBounds.width / 2 - tooltipEl.offsetWidth / 2;
|
||||||
const tooltipWidth = tooltipRef.current.offsetWidth;
|
setPos(calcTooltipPosition({ x: offsetX, y: offsetY }));
|
||||||
const tooltipHeight = tooltipRef.current.offsetHeight;
|
return;
|
||||||
let newLeft = clientX + 10;
|
|
||||||
let newTop = clientY + 10;
|
|
||||||
if (newLeft + tooltipWidth + 10 > window.innerWidth) {
|
|
||||||
newLeft = window.innerWidth - tooltipWidth - 10;
|
|
||||||
}
|
|
||||||
if (newTop + tooltipHeight + 10 > window.innerHeight) {
|
|
||||||
newTop = window.innerHeight - tooltipHeight - 10;
|
|
||||||
}
|
|
||||||
setPosition({ top: newTop, left: newLeft });
|
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
const deb = debounce(handleMouseMove, 10);
|
setPos(calcTooltipPosition({ x: offsetX, y: offsetY }));
|
||||||
|
}, [calcTooltipPosition, ev, tPosition, offset]);
|
||||||
|
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
useEffect(() => {
|
||||||
// @ts-expect-error
|
if (!targetSelector) return;
|
||||||
document.addEventListener('mousemove', deb);
|
|
||||||
return () => {
|
|
||||||
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
||||||
// @ts-expect-error
|
|
||||||
document.removeEventListener('mousemove', deb);
|
|
||||||
};
|
|
||||||
}, [targetSelector]);
|
|
||||||
|
|
||||||
return createPortal(
|
function handleMouseMove(nativeEvt: globalThis.MouseEvent) {
|
||||||
visible && (
|
const targetEl = nativeEvt.target as HTMLElement | null;
|
||||||
<div
|
if (!targetEl) {
|
||||||
ref={tooltipRef}
|
setVisible(false);
|
||||||
className={clsx(
|
return;
|
||||||
classes.tooltip,
|
}
|
||||||
'pointer-events-none',
|
const triggerEl = targetEl.closest(targetSelector!);
|
||||||
'absolute px-2 py-2',
|
const isInsideTooltip = interactive && tooltipRef.current?.contains(targetEl);
|
||||||
'border rounded border-green-300 border-opacity-10 bg-stone-900 bg-opacity-90',
|
|
||||||
{ ['invisible']: position === null },
|
if (!triggerEl && !isInsideTooltip) {
|
||||||
className,
|
setVisible(false);
|
||||||
)}
|
return;
|
||||||
style={{
|
}
|
||||||
top: position?.top ?? 0,
|
setVisible(true);
|
||||||
left: position?.left ?? 0,
|
|
||||||
zIndex: 10000,
|
if (triggerEl && tooltipRef.current) {
|
||||||
}}
|
const rect = triggerEl.getBoundingClientRect();
|
||||||
>
|
const tooltipEl = tooltipRef.current;
|
||||||
{typeof content === 'function' ? content() : content}
|
let x = nativeEvt.clientX;
|
||||||
</div>
|
let y = nativeEvt.clientY;
|
||||||
),
|
|
||||||
document.body,
|
if (tPosition === TooltipPosition.left) {
|
||||||
);
|
x = rect.left - tooltipEl.offsetWidth - offset;
|
||||||
});
|
y = rect.y + rect.height / 2 - tooltipEl.offsetHeight / 2;
|
||||||
|
if (x <= 0) {
|
||||||
|
x = rect.left + rect.width + offset;
|
||||||
|
}
|
||||||
|
} else if (tPosition === TooltipPosition.right) {
|
||||||
|
x = rect.left + rect.width + offset;
|
||||||
|
y = rect.y + rect.height / 2 - tooltipEl.offsetHeight / 2;
|
||||||
|
} else if (tPosition === TooltipPosition.top) {
|
||||||
|
x = rect.x + rect.width / 2 - tooltipEl.offsetWidth / 2;
|
||||||
|
y = rect.top - tooltipEl.offsetHeight - offset;
|
||||||
|
} else if (tPosition === TooltipPosition.bottom) {
|
||||||
|
x = rect.x + rect.width / 2 - tooltipEl.offsetWidth / 2;
|
||||||
|
y = rect.bottom + offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
setPos(calcTooltipPosition({ x, y }));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const debounced = debounce(handleMouseMove, 10);
|
||||||
|
|
||||||
|
const listener: EventListener = evt => {
|
||||||
|
debounced(evt as globalThis.MouseEvent);
|
||||||
|
};
|
||||||
|
|
||||||
|
document.addEventListener('mousemove', listener);
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener('mousemove', listener);
|
||||||
|
};
|
||||||
|
}, [targetSelector, interactive, tPosition, offset, calcTooltipPosition]);
|
||||||
|
|
||||||
|
return createPortal(
|
||||||
|
visible && (
|
||||||
|
<div
|
||||||
|
ref={tooltipRef}
|
||||||
|
className={clsx(
|
||||||
|
classes.tooltip,
|
||||||
|
interactive ? 'pointer-events-auto' : 'pointer-events-none',
|
||||||
|
'absolute p-1 border rounded-sm border-green-300 border-opacity-10 bg-stone-900 bg-opacity-90',
|
||||||
|
pos === null ? 'invisible' : '',
|
||||||
|
className,
|
||||||
|
)}
|
||||||
|
style={{
|
||||||
|
top: pos?.top ?? 0,
|
||||||
|
left: pos?.left ?? 0,
|
||||||
|
zIndex: 10000,
|
||||||
|
}}
|
||||||
|
onMouseEnter={() => interactive && setIsMouseInsideTooltip(true)}
|
||||||
|
onMouseLeave={() => interactive && setIsMouseInsideTooltip(false)}
|
||||||
|
>
|
||||||
|
{typeof content === 'function' ? content() : content}
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
document.body,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
WdTooltip.displayName = 'WdTooltip';
|
||||||
|
|||||||
@@ -1,49 +1,57 @@
|
|||||||
import React, { HTMLProps, MouseEventHandler, useCallback, useRef } from 'react';
|
import { forwardRef, HTMLProps, ReactNode } from 'react';
|
||||||
|
|
||||||
import classes from './WdTooltipWrapper.module.scss';
|
|
||||||
import { WithChildren, WithClassName } from '@/hooks/Mapper/types/common.ts';
|
|
||||||
import { TooltipProps, WdTooltip, WdTooltipHandlers } from '@/hooks/Mapper/components/ui-kit';
|
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
|
import { WdTooltip, WdTooltipHandlers, TooltipProps } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import classes from './WdTooltipWrapper.module.scss';
|
||||||
|
|
||||||
|
type TooltipSize = 'xs' | 'sm' | 'md' | 'lg';
|
||||||
|
|
||||||
export type WdTooltipWrapperProps = {
|
export type WdTooltipWrapperProps = {
|
||||||
content?: (() => React.ReactNode) | React.ReactNode;
|
content?: (() => ReactNode) | ReactNode;
|
||||||
} & WithChildren &
|
size?: TooltipSize;
|
||||||
WithClassName &
|
interactive?: boolean;
|
||||||
HTMLProps<HTMLDivElement> &
|
} & Omit<HTMLProps<HTMLDivElement>, 'content' | 'size'> &
|
||||||
Omit<TooltipProps, 'content'>;
|
Omit<TooltipProps, 'content'>;
|
||||||
|
|
||||||
export const WdTooltipWrapper = ({
|
export const WdTooltipWrapper = forwardRef<WdTooltipHandlers, WdTooltipWrapperProps>(
|
||||||
className,
|
(
|
||||||
children,
|
{ className, children, content, offset, position, targetSelector, interactive = false, size, ...props },
|
||||||
content,
|
forwardedRef,
|
||||||
offset,
|
) => {
|
||||||
position,
|
const suffix = Math.random().toString(36).slice(2, 7);
|
||||||
targetSelector,
|
const autoClass = `wdTooltipAutoTrigger-${suffix}`;
|
||||||
...props
|
const finalTargetSelector = targetSelector || `.${autoClass}`;
|
||||||
}: WdTooltipWrapperProps) => {
|
|
||||||
const tooltipRef = useRef<WdTooltipHandlers>(null);
|
|
||||||
const handleShowDeleteTooltip: MouseEventHandler = useCallback(e => tooltipRef.current?.show(e), []);
|
|
||||||
const handleHideDeleteTooltip: MouseEventHandler = useCallback(e => tooltipRef.current?.hide(e), []);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<div className={clsx(classes.WdTooltipWrapperRoot, className)} {...props}>
|
||||||
<div
|
{targetSelector ? <>{children}</> : <div className={autoClass}>{children}</div>}
|
||||||
className={clsx(classes.WdTooltipWrapperRoot, className)}
|
|
||||||
{...props}
|
<WdTooltip
|
||||||
{...(content && {
|
ref={forwardedRef}
|
||||||
onMouseEnter: handleShowDeleteTooltip,
|
offset={offset}
|
||||||
onMouseLeave: handleHideDeleteTooltip,
|
position={position}
|
||||||
})}
|
content={content}
|
||||||
>
|
interactive={interactive}
|
||||||
{children}
|
targetSelector={finalTargetSelector}
|
||||||
|
className={size ? sizeClass(size) : undefined}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<WdTooltip
|
);
|
||||||
ref={tooltipRef}
|
},
|
||||||
offset={offset}
|
);
|
||||||
position={position}
|
|
||||||
content={content}
|
WdTooltipWrapper.displayName = 'WdTooltipWrapper';
|
||||||
targetSelector={targetSelector}
|
|
||||||
/>
|
function sizeClass(size: TooltipSize) {
|
||||||
</>
|
switch (size) {
|
||||||
);
|
case 'xs':
|
||||||
};
|
return classes.wdTooltipSizeXs;
|
||||||
|
case 'sm':
|
||||||
|
return classes.wdTooltipSizeSm;
|
||||||
|
case 'md':
|
||||||
|
return classes.wdTooltipSizeMd;
|
||||||
|
case 'lg':
|
||||||
|
return classes.wdTooltipSizeLg;
|
||||||
|
default:
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -0,0 +1,137 @@
|
|||||||
|
.windowContainer {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.window {
|
||||||
|
position: absolute;
|
||||||
|
//background: #fff;
|
||||||
|
//border: 1px solid #000;
|
||||||
|
//box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
|
||||||
|
user-select: none;
|
||||||
|
|
||||||
|
pointer-events: initial;
|
||||||
|
}
|
||||||
|
|
||||||
|
.resizeHandle {
|
||||||
|
position: absolute;
|
||||||
|
//background: rgba(0, 0, 0, 0.2);
|
||||||
|
width: 15px;
|
||||||
|
height: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.topRight,
|
||||||
|
.bottomLeft {
|
||||||
|
cursor: nesw-resize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.topLeft,
|
||||||
|
.bottomRight {
|
||||||
|
cursor: nwse-resize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.topLeft {
|
||||||
|
top: -7.5px;
|
||||||
|
left: -7.5px;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
position: relative;
|
||||||
|
top: 7.5px;
|
||||||
|
left: 7.5px;
|
||||||
|
display: block;
|
||||||
|
content: " ";
|
||||||
|
width: 5px;
|
||||||
|
height: 5px;
|
||||||
|
border-left: 1px solid var(--window-corner);
|
||||||
|
border-top: 1px solid var(--window-corner);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.topRight {
|
||||||
|
top: -7.5px;
|
||||||
|
right: -7.5px;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
position: relative;
|
||||||
|
top: 7.5px;
|
||||||
|
right: -2.5px;
|
||||||
|
display: block;
|
||||||
|
content: " ";
|
||||||
|
width: 5px;
|
||||||
|
height: 5px;
|
||||||
|
border-right: 1px solid var(--window-corner);
|
||||||
|
border-top: 1px solid var(--window-corner);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.bottomLeft {
|
||||||
|
bottom: -7.5px;
|
||||||
|
left: -7.5px;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
position: relative;
|
||||||
|
top: 2.5px;
|
||||||
|
left: 7.5px;
|
||||||
|
display: block;
|
||||||
|
content: " ";
|
||||||
|
width: 5px;
|
||||||
|
height: 5px;
|
||||||
|
border-left: 1px solid var(--window-corner);
|
||||||
|
border-bottom: 1px solid var(--window-corner);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.bottomRight {
|
||||||
|
bottom: -7.5px;
|
||||||
|
right: -7.5px;
|
||||||
|
|
||||||
|
&::after {
|
||||||
|
position: relative;
|
||||||
|
top: 2.5px;
|
||||||
|
right: -2.5px;
|
||||||
|
display: block;
|
||||||
|
content: " ";
|
||||||
|
width: 5px;
|
||||||
|
height: 5px;
|
||||||
|
border-right: 1px solid var(--window-corner);
|
||||||
|
border-bottom: 1px solid var(--window-corner);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.top {
|
||||||
|
top: -5px;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
height: 10px;
|
||||||
|
cursor: ns-resize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bottom {
|
||||||
|
bottom: -5px;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
height: 10px;
|
||||||
|
cursor: ns-resize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left {
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: -5px;
|
||||||
|
width: 10px;
|
||||||
|
cursor: ew-resize;
|
||||||
|
}
|
||||||
|
|
||||||
|
.right {
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
right: -5px;
|
||||||
|
width: 10px;
|
||||||
|
cursor: ew-resize;
|
||||||
|
}
|
||||||
@@ -0,0 +1,423 @@
|
|||||||
|
import React, { useState, useRef, useEffect, useMemo, useCallback } from 'react';
|
||||||
|
import styles from './WindowManager.module.scss';
|
||||||
|
import debounce from 'lodash.debounce';
|
||||||
|
import { WindowProps } from '@/hooks/Mapper/components/ui-kit/WindowManager/types.ts';
|
||||||
|
|
||||||
|
const MIN_WINDOW_SIZE = 100;
|
||||||
|
const SNAP_THRESHOLD = 10;
|
||||||
|
export const SNAP_GAP = 10;
|
||||||
|
|
||||||
|
export enum ActionType {
|
||||||
|
Drag = 'drag',
|
||||||
|
Resize = 'resize',
|
||||||
|
}
|
||||||
|
|
||||||
|
export const DefaultWindowState = {
|
||||||
|
x: 0,
|
||||||
|
y: 0,
|
||||||
|
width: 0,
|
||||||
|
height: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
function getWindowsBySides(windows: WindowProps[], containerWidth: number, containerHeight: number) {
|
||||||
|
const centerX = containerWidth / 2;
|
||||||
|
const centerY = containerHeight / 2;
|
||||||
|
|
||||||
|
const top = windows.filter(window => window.position.y + window.size.height / 2 < centerY);
|
||||||
|
const bottom = windows.filter(window => window.position.y + window.size.height / 2 >= centerY);
|
||||||
|
const left = windows.filter(window => window.position.x + window.size.width / 2 < centerX);
|
||||||
|
const right = windows.filter(window => window.position.x + window.size.width / 2 >= centerX);
|
||||||
|
|
||||||
|
return { top, bottom, left, right };
|
||||||
|
}
|
||||||
|
|
||||||
|
export type WindowWrapperProps = {
|
||||||
|
onDrag: (e: React.MouseEvent, windowId: string | number) => void;
|
||||||
|
onResize: (e: React.MouseEvent, windowId: string | number, resizeDirection: string) => void;
|
||||||
|
} & WindowProps;
|
||||||
|
|
||||||
|
export const WindowWrapper = ({ onResize, onDrag, ...window }: WindowWrapperProps) => {
|
||||||
|
const handleMouseDownRoot = (e: React.MouseEvent) => {
|
||||||
|
onDrag(e, window.id);
|
||||||
|
};
|
||||||
|
|
||||||
|
const { handleResizeTL, handleResizeTR, handleResizeBL, handleResizeBR } = useMemo(() => {
|
||||||
|
const handleResizeTL = (e: React.MouseEvent) => onResize(e, window.id, 'top left');
|
||||||
|
const handleResizeTR = (e: React.MouseEvent) => onResize(e, window.id, 'top right');
|
||||||
|
const handleResizeBL = (e: React.MouseEvent) => onResize(e, window.id, 'bottom left');
|
||||||
|
const handleResizeBR = (e: React.MouseEvent) => onResize(e, window.id, 'bottom right');
|
||||||
|
|
||||||
|
return {
|
||||||
|
handleResizeTL,
|
||||||
|
handleResizeTR,
|
||||||
|
handleResizeBL,
|
||||||
|
handleResizeBR,
|
||||||
|
};
|
||||||
|
}, [window]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={window.id}
|
||||||
|
className={`drag-handle ${styles.window}`}
|
||||||
|
style={{
|
||||||
|
width: window.size.width,
|
||||||
|
height: window.size.height,
|
||||||
|
top: window.position.y,
|
||||||
|
left: window.position.x,
|
||||||
|
zIndex: window.zIndex,
|
||||||
|
}}
|
||||||
|
onMouseDown={handleMouseDownRoot}
|
||||||
|
>
|
||||||
|
{window.content(window)}
|
||||||
|
<div className={styles.resizeHandle + ' ' + styles.topLeft} onMouseDown={handleResizeTL} />
|
||||||
|
<div className={styles.resizeHandle + ' ' + styles.topRight} onMouseDown={handleResizeTR} />
|
||||||
|
<div className={styles.resizeHandle + ' ' + styles.bottomLeft} onMouseDown={handleResizeBL} />
|
||||||
|
<div className={styles.resizeHandle + ' ' + styles.bottomRight} onMouseDown={handleResizeBR} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
type WindowManagerProps = {
|
||||||
|
windows: WindowProps[];
|
||||||
|
dragSelector?: string;
|
||||||
|
onChange?(windows: WindowProps[]): void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WindowManager: React.FC<WindowManagerProps> = ({ windows: initialWindows, dragSelector, onChange }) => {
|
||||||
|
const [windows, setWindows] = useState(
|
||||||
|
initialWindows.map((window, index) => ({
|
||||||
|
...window,
|
||||||
|
zIndex: index + 1,
|
||||||
|
})),
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setWindows(initialWindows.slice(0));
|
||||||
|
}, [initialWindows]);
|
||||||
|
|
||||||
|
const containerRef = useRef<HTMLDivElement | null>(null);
|
||||||
|
const activeWindowIdRef = useRef<string | number | null>(null);
|
||||||
|
const actionTypeRef = useRef<ActionType | null>(null);
|
||||||
|
const resizeDirectionRef = useRef<string | null>(null);
|
||||||
|
const startMousePositionRef = useRef<{ x: number; y: number }>({ x: 0, y: 0 });
|
||||||
|
const startWindowStateRef = useRef<{ x: number; y: number; width: number; height: number }>(DefaultWindowState);
|
||||||
|
|
||||||
|
const ref = useRef({ windows, onChange });
|
||||||
|
ref.current = { windows, onChange };
|
||||||
|
|
||||||
|
const refPrevSize = useRef({ w: 0, h: 0 });
|
||||||
|
|
||||||
|
const onDebouncedChange = useMemo(() => {
|
||||||
|
return debounce(() => {
|
||||||
|
ref.current.onChange?.(ref.current.windows);
|
||||||
|
}, 20);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleMouseDown = (
|
||||||
|
e: React.MouseEvent,
|
||||||
|
windowId: string | number,
|
||||||
|
actionType: ActionType,
|
||||||
|
resizeDirection?: string,
|
||||||
|
) => {
|
||||||
|
if (dragSelector && actionType === ActionType.Drag && !(e.target as HTMLElement).closest(dragSelector)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
e.stopPropagation();
|
||||||
|
activeWindowIdRef.current = windowId;
|
||||||
|
actionTypeRef.current = actionType;
|
||||||
|
resizeDirectionRef.current = resizeDirection || null;
|
||||||
|
startMousePositionRef.current = { x: e.clientX, y: e.clientY };
|
||||||
|
const targetWindow = windows.find(win => win.id === windowId);
|
||||||
|
if (targetWindow) {
|
||||||
|
startWindowStateRef.current = {
|
||||||
|
x: targetWindow.position.x,
|
||||||
|
y: targetWindow.position.y,
|
||||||
|
width: targetWindow.size.width,
|
||||||
|
height: targetWindow.size.height,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bring window to front by updating zIndex
|
||||||
|
setWindows(prevWindows => {
|
||||||
|
const maxZIndex = Math.max(...prevWindows.map(w => w.zIndex));
|
||||||
|
return prevWindows.map(window => (window.id === windowId ? { ...window, zIndex: maxZIndex + 1 } : window));
|
||||||
|
});
|
||||||
|
|
||||||
|
window.addEventListener('mousemove', handleMouseMove);
|
||||||
|
window.addEventListener('mouseup', handleMouseUp);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleMouseMove = (e: MouseEvent) => {
|
||||||
|
if (activeWindowIdRef.current !== null && actionTypeRef.current) {
|
||||||
|
const deltaX = e.clientX - startMousePositionRef.current.x;
|
||||||
|
const deltaY = e.clientY - startMousePositionRef.current.y;
|
||||||
|
const container = containerRef.current;
|
||||||
|
|
||||||
|
setWindows(prevWindows =>
|
||||||
|
prevWindows.map(window => {
|
||||||
|
if (window.id === activeWindowIdRef.current) {
|
||||||
|
let newX = startWindowStateRef.current.x;
|
||||||
|
let newY = startWindowStateRef.current.y;
|
||||||
|
let newWidth = startWindowStateRef.current.width;
|
||||||
|
let newHeight = startWindowStateRef.current.height;
|
||||||
|
|
||||||
|
if (actionTypeRef.current === ActionType.Drag) {
|
||||||
|
newX += deltaX;
|
||||||
|
newY += deltaY;
|
||||||
|
|
||||||
|
// Ensure the window stays within the container boundaries
|
||||||
|
if (container) {
|
||||||
|
newX = Math.max(SNAP_GAP, Math.min(container.clientWidth - window.size.width - SNAP_GAP, newX));
|
||||||
|
newY = Math.max(SNAP_GAP, Math.min(container.clientHeight - window.size.height - SNAP_GAP, newY));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Snap to other windows with or without gap
|
||||||
|
prevWindows.forEach(otherWindow => {
|
||||||
|
if (otherWindow.id === window.id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Snap vertically (top and bottom)
|
||||||
|
if (Math.abs(newY - otherWindow.position.y) < SNAP_THRESHOLD) {
|
||||||
|
newY = otherWindow.position.y; // Align top without gap
|
||||||
|
} else if (Math.abs(newY + window.size.height - otherWindow.position.y) < SNAP_THRESHOLD) {
|
||||||
|
newY = otherWindow.position.y - window.size.height - SNAP_GAP; // Bottom aligns to top
|
||||||
|
} else if (Math.abs(newY - (otherWindow.position.y + otherWindow.size.height)) < SNAP_THRESHOLD) {
|
||||||
|
newY = otherWindow.position.y + otherWindow.size.height + SNAP_GAP; // Align bottom without gap
|
||||||
|
} else if (
|
||||||
|
Math.abs(newY + window.size.height - (otherWindow.position.y + otherWindow.size.height)) <
|
||||||
|
SNAP_THRESHOLD
|
||||||
|
) {
|
||||||
|
newY = otherWindow.position.y + otherWindow.size.height - window.size.height; // Bottom aligns bottom
|
||||||
|
}
|
||||||
|
|
||||||
|
// Snap horizontally (left and right)
|
||||||
|
if (Math.abs(newX - otherWindow.position.x) < SNAP_THRESHOLD) {
|
||||||
|
newX = otherWindow.position.x; // Align left without gap
|
||||||
|
} else if (Math.abs(newX + window.size.width - otherWindow.position.x) < SNAP_THRESHOLD) {
|
||||||
|
newX = otherWindow.position.x - window.size.width - SNAP_GAP; // Right aligns to left
|
||||||
|
} else if (Math.abs(newX - (otherWindow.position.x + otherWindow.size.width)) < SNAP_THRESHOLD) {
|
||||||
|
newX = otherWindow.position.x + otherWindow.size.width + SNAP_GAP; // Align right without gap
|
||||||
|
} else if (
|
||||||
|
Math.abs(newX + window.size.width - (otherWindow.position.x + otherWindow.size.width)) <
|
||||||
|
SNAP_THRESHOLD
|
||||||
|
) {
|
||||||
|
newX = otherWindow.position.x + otherWindow.size.width - window.size.width; // Right aligns right
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (actionTypeRef.current === ActionType.Resize && resizeDirectionRef.current) {
|
||||||
|
if (resizeDirectionRef.current.includes('right')) {
|
||||||
|
newWidth = Math.max(MIN_WINDOW_SIZE, startWindowStateRef.current.width + deltaX);
|
||||||
|
|
||||||
|
// Снап для правой границы с отступом SNAP_THRESHOLD
|
||||||
|
prevWindows.forEach(otherWindow => {
|
||||||
|
if (otherWindow.id !== window.id) {
|
||||||
|
// Правая граница текущего окна к левой границе другого окна
|
||||||
|
const snapRightToLeft =
|
||||||
|
otherWindow.position.x - (startWindowStateRef.current.x + newWidth) - SNAP_THRESHOLD;
|
||||||
|
if (Math.abs(snapRightToLeft) < SNAP_THRESHOLD) {
|
||||||
|
newWidth = otherWindow.position.x - startWindowStateRef.current.x - SNAP_THRESHOLD;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Правая граница текущего окна к правой границе другого окна
|
||||||
|
const snapRightToRight =
|
||||||
|
otherWindow.position.x + otherWindow.size.width - (startWindowStateRef.current.x + newWidth);
|
||||||
|
if (Math.abs(snapRightToRight) < SNAP_THRESHOLD) {
|
||||||
|
newWidth = otherWindow.position.x + otherWindow.size.width - startWindowStateRef.current.x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resizeDirectionRef.current.includes('left')) {
|
||||||
|
newWidth = Math.max(MIN_WINDOW_SIZE, startWindowStateRef.current.width - deltaX);
|
||||||
|
newX = startWindowStateRef.current.x + (startWindowStateRef.current.width - newWidth);
|
||||||
|
|
||||||
|
// Снап для левой границы с отступом SNAP_THRESHOLD
|
||||||
|
prevWindows.forEach(otherWindow => {
|
||||||
|
if (otherWindow.id !== window.id) {
|
||||||
|
// Левая граница текущего окна к правой границе другого окна
|
||||||
|
const snapLeftToRight = newX - (otherWindow.position.x + otherWindow.size.width + SNAP_THRESHOLD);
|
||||||
|
if (Math.abs(snapLeftToRight) < SNAP_THRESHOLD) {
|
||||||
|
newX = otherWindow.position.x + otherWindow.size.width + SNAP_THRESHOLD;
|
||||||
|
newWidth = startWindowStateRef.current.width + startWindowStateRef.current.x - newX;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Левая граница текущего окна к левой границе другого окна
|
||||||
|
const snapLeftToLeft = newX - otherWindow.position.x;
|
||||||
|
if (Math.abs(snapLeftToLeft) < SNAP_THRESHOLD) {
|
||||||
|
newX = otherWindow.position.x;
|
||||||
|
newWidth = startWindowStateRef.current.width + startWindowStateRef.current.x - newX;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resizeDirectionRef.current.includes('bottom')) {
|
||||||
|
newHeight = Math.max(MIN_WINDOW_SIZE, startWindowStateRef.current.height + deltaY);
|
||||||
|
|
||||||
|
// Снап для нижней границы с отступом SNAP_THRESHOLD
|
||||||
|
prevWindows.forEach(otherWindow => {
|
||||||
|
if (otherWindow.id !== window.id) {
|
||||||
|
// Нижняя граница текущего окна к верхней границе другого окна
|
||||||
|
const snapBottomToTop =
|
||||||
|
otherWindow.position.y - (startWindowStateRef.current.y + newHeight) - SNAP_THRESHOLD;
|
||||||
|
if (Math.abs(snapBottomToTop) < SNAP_THRESHOLD) {
|
||||||
|
newHeight = otherWindow.position.y - startWindowStateRef.current.y - SNAP_THRESHOLD;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Нижняя граница текущего окна к нижней границе другого окна
|
||||||
|
const snapBottomToBottom =
|
||||||
|
otherWindow.position.y + otherWindow.size.height - (startWindowStateRef.current.y + newHeight);
|
||||||
|
if (Math.abs(snapBottomToBottom) < SNAP_THRESHOLD) {
|
||||||
|
newHeight = otherWindow.position.y + otherWindow.size.height - startWindowStateRef.current.y;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resizeDirectionRef.current.includes('top')) {
|
||||||
|
newHeight = Math.max(MIN_WINDOW_SIZE, startWindowStateRef.current.height - deltaY);
|
||||||
|
newY = startWindowStateRef.current.y + (startWindowStateRef.current.height - newHeight);
|
||||||
|
|
||||||
|
// Снап для верхней границы с отступом SNAP_THRESHOLD
|
||||||
|
prevWindows.forEach(otherWindow => {
|
||||||
|
if (otherWindow.id !== window.id) {
|
||||||
|
// Верхняя граница текущего окна к нижней границе другого окна
|
||||||
|
const snapTopToBottom = newY - (otherWindow.position.y + otherWindow.size.height + SNAP_THRESHOLD);
|
||||||
|
if (Math.abs(snapTopToBottom) < SNAP_THRESHOLD) {
|
||||||
|
newY = otherWindow.position.y + otherWindow.size.height + SNAP_THRESHOLD;
|
||||||
|
newHeight = startWindowStateRef.current.height + startWindowStateRef.current.y - newY;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Верхняя граница текущего окна к верхней границе другого окна
|
||||||
|
const snapTopToTop = newY - otherWindow.position.y;
|
||||||
|
if (Math.abs(snapTopToTop) < SNAP_THRESHOLD) {
|
||||||
|
newY = otherWindow.position.y;
|
||||||
|
newHeight = startWindowStateRef.current.height + startWindowStateRef.current.y - newY;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure the window stays within the container boundaries
|
||||||
|
if (container) {
|
||||||
|
newX = Math.max(0 + SNAP_GAP, Math.min(container.clientWidth - newWidth - SNAP_GAP, newX));
|
||||||
|
newY = Math.max(0 + SNAP_GAP, Math.min(container.clientHeight - newHeight - SNAP_GAP, newY));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...window,
|
||||||
|
position: { x: newX, y: newY },
|
||||||
|
size: { width: newWidth, height: newHeight },
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return window;
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
onDebouncedChange();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleMouseUp = useCallback(() => {
|
||||||
|
activeWindowIdRef.current = null;
|
||||||
|
actionTypeRef.current = null;
|
||||||
|
resizeDirectionRef.current = null;
|
||||||
|
|
||||||
|
onDebouncedChange();
|
||||||
|
window.removeEventListener('mousemove', handleMouseMove);
|
||||||
|
window.removeEventListener('mouseup', handleMouseUp);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// Handle resize of the container and reposition windows
|
||||||
|
useEffect(() => {
|
||||||
|
if (containerRef.current) {
|
||||||
|
refPrevSize.current = { w: containerRef.current.clientWidth, h: containerRef.current.clientHeight };
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleResize = () => {
|
||||||
|
const container = containerRef.current;
|
||||||
|
const { windows } = ref.current;
|
||||||
|
if (!container) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const deltaX = container.clientWidth - refPrevSize.current.w;
|
||||||
|
const deltaY = container.clientHeight - refPrevSize.current.h;
|
||||||
|
|
||||||
|
const { bottom, right } = getWindowsBySides(windows, refPrevSize.current.w, refPrevSize.current.h);
|
||||||
|
|
||||||
|
setWindows(w => {
|
||||||
|
return w.map(x => {
|
||||||
|
let next = { ...x };
|
||||||
|
|
||||||
|
if (right.some(r => r.id === x.id)) {
|
||||||
|
next = {
|
||||||
|
...next,
|
||||||
|
position: {
|
||||||
|
...next.position,
|
||||||
|
x: next.position.x + deltaX,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bottom.some(r => r.id === x.id)) {
|
||||||
|
next = {
|
||||||
|
...next,
|
||||||
|
position: {
|
||||||
|
...next.position,
|
||||||
|
y: next.position.y + deltaY,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next.position.x + next.size.width > container.clientWidth - SNAP_GAP) {
|
||||||
|
next.position.x = container.clientWidth - next.size.width - SNAP_GAP;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next.position.y + next.size.height > container.clientHeight - SNAP_GAP) {
|
||||||
|
next.position.y = container.clientHeight - next.size.height - SNAP_GAP;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (next.position.y < 0) {
|
||||||
|
next.position.y = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return next;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
onDebouncedChange();
|
||||||
|
|
||||||
|
refPrevSize.current = { w: container.clientWidth, h: container.clientHeight };
|
||||||
|
};
|
||||||
|
|
||||||
|
const tid = setTimeout(handleResize, 10);
|
||||||
|
window.addEventListener('resize', handleResize);
|
||||||
|
return () => {
|
||||||
|
clearTimeout(tid);
|
||||||
|
window.removeEventListener('resize', handleResize);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleDrag = (e: React.MouseEvent, windowId: string | number) => {
|
||||||
|
handleMouseDown(e, windowId, ActionType.Drag);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleResize = (e: React.MouseEvent, windowId: string | number, resizeDirection: string) => {
|
||||||
|
handleMouseDown(e, windowId, ActionType.Resize, resizeDirection);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div ref={containerRef} className={styles.windowContainer}>
|
||||||
|
{windows.map(window => (
|
||||||
|
<WindowWrapper key={window.id} onDrag={handleDrag} onResize={handleResize} {...window} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './WindowManager';
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
import React from 'react';
|
||||||
|
|
||||||
|
export type WindowProps = {
|
||||||
|
id: string | number;
|
||||||
|
content: (w: WindowProps) => React.ReactNode;
|
||||||
|
position: { x: number; y: number };
|
||||||
|
size: { width: number; height: number };
|
||||||
|
zIndex: number;
|
||||||
|
visible?: boolean;
|
||||||
|
};
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
import { WindowProps } from '@/hooks/Mapper/components/ui-kit/WindowManager/WindowManager.tsx';
|
||||||
|
|
||||||
|
export function getWindowsBySides(windows: WindowProps[], containerWidth: number, containerHeight: number) {
|
||||||
|
const centerX = containerWidth / 2;
|
||||||
|
const centerY = containerHeight / 2;
|
||||||
|
|
||||||
|
const top = windows.filter(window => window.position.y + window.size.height / 2 < centerY);
|
||||||
|
const bottom = windows.filter(window => window.position.y + window.size.height / 2 >= centerY);
|
||||||
|
const left = windows.filter(window => window.position.x + window.size.width / 2 < centerX);
|
||||||
|
const right = windows.filter(window => window.position.x + window.size.width / 2 >= centerX);
|
||||||
|
|
||||||
|
return { top, bottom, left, right };
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user