mirror of
https://github.com/wanderer-industries/wanderer
synced 2025-12-06 15:55:36 +00:00
Compare commits
150 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
f6db6f0914 | ||
|
|
ab8baeedd1 | ||
|
|
eccee5e72e | ||
|
|
4d93055bda | ||
|
|
c60c16e56a | ||
|
|
99b1de5647 | ||
|
|
7efe11a421 | ||
|
|
954108856a | ||
|
|
cbca745ec4 | ||
|
|
e15e7c8f8d | ||
|
|
65e8a520e5 | ||
|
|
3926af5a6d | ||
|
|
556fb33223 | ||
|
|
82295adeab | ||
|
|
efabf060c7 | ||
|
|
96e434ebf5 | ||
|
|
d81e2567cc | ||
|
|
f8d487639f | ||
|
|
cecfbb5375 | ||
|
|
8d35500e2f | ||
|
|
5dad5d8e03 | ||
|
|
9d7d4fad2e | ||
|
|
7be64bde02 | ||
|
|
48eb7552a9 | ||
|
|
5347b0060c | ||
|
|
b826c03226 | ||
|
|
1c211a8667 | ||
|
|
fd4d5b90e2 | ||
|
|
1ee9f26b34 | ||
|
|
da1762934b | ||
|
|
511457c761 | ||
|
|
29b4cedb81 | ||
|
|
585de15e6b | ||
|
|
a9bf118f3a | ||
|
|
6d5a432bad | ||
|
|
f1f12abd16 | ||
|
|
09880a54e9 | ||
|
|
0f6847b16d | ||
|
|
ce82ed97f5 | ||
|
|
36b393dbde | ||
|
|
524c283a0d | ||
|
|
afda53a9bc | ||
|
|
1310d75012 | ||
|
|
80bbde549d | ||
|
|
2451487593 | ||
|
|
ecd626f105 | ||
|
|
123b312965 | ||
|
|
e94de8e629 | ||
|
|
956a5a04ca | ||
|
|
affeb7c624 | ||
|
|
e457d94df8 | ||
|
|
e9583c928e | ||
|
|
89c14628e1 | ||
|
|
ffba407eaf | ||
|
|
33f710127c | ||
|
|
7a82b2c102 | ||
|
|
2db2a47186 | ||
|
|
63faa43c1d | ||
|
|
eabb0e8470 | ||
|
|
9f75ae6b03 | ||
|
|
a1f28cd245 | ||
|
|
90a04b517e | ||
|
|
9f6e6a333f | ||
|
|
7b9e2c4fd9 | ||
|
|
63f13711cc | ||
|
|
c65b8e5ebd | ||
|
|
bfed1480ae | ||
|
|
5ff902f185 | ||
|
|
8d38345c7f | ||
|
|
14be9dbb8a | ||
|
|
720c26db94 | ||
|
|
6d0b8b845d | ||
|
|
b2767e000e | ||
|
|
169f23c2ca | ||
|
|
81f70eafff | ||
|
|
650170498a | ||
|
|
8b6f600989 | ||
|
|
fe3617b39f | ||
|
|
0f466c51ba | ||
|
|
a1a641bce3 | ||
|
|
4764c25eb1 | ||
|
|
d390455cf2 | ||
|
|
7fb8d24d73 | ||
|
|
472dbaa68b | ||
|
|
f03448007d | ||
|
|
c317a8bff9 | ||
|
|
618cca39a4 | ||
|
|
fe7a98098f | ||
|
|
df49939990 | ||
|
|
f23f2776f4 | ||
|
|
4419c86164 | ||
|
|
9848f49b49 | ||
|
|
679bd782a8 | ||
|
|
6a316e3906 | ||
|
|
c129db8474 | ||
|
|
10035b4c91 | ||
|
|
5839271de7 | ||
|
|
47db8ef709 | ||
|
|
2656491aaa | ||
|
|
a7637c9cae | ||
|
|
7b83ed8205 | ||
|
|
00cbc77f1d | ||
|
|
4d75b256c4 | ||
|
|
5aeff7c40c | ||
|
|
6a543bf644 | ||
|
|
dfb035525d | ||
|
|
4c23069a0a | ||
|
|
4a1d7be44c | ||
|
|
798aec1b74 | ||
|
|
7914d7e151 | ||
|
|
26d0392da1 | ||
|
|
83b1406cce | ||
|
|
8b579d6837 | ||
|
|
c0fd20dfff | ||
|
|
dd6b67c6e6 | ||
|
|
fa83185cf5 | ||
|
|
97d5010d41 | ||
|
|
e73ad93920 | ||
|
|
425af246fb | ||
|
|
a2912ba0ff | ||
|
|
48ff2f4413 | ||
|
|
61cd281a18 | ||
|
|
6e28134282 | ||
|
|
d1377f44d2 | ||
|
|
d261c6186b | ||
|
|
2a72a2612d | ||
|
|
66bb4f87d4 | ||
|
|
977b1ad083 | ||
|
|
94db18d42b | ||
|
|
7e0375108d | ||
|
|
094a5d7b62 | ||
|
|
8f947a5f04 | ||
|
|
5580ad62f9 | ||
|
|
c0953dc954 | ||
|
|
1df93da564 | ||
|
|
e2252a9d72 | ||
|
|
7cdba4b507 | ||
|
|
b110d5afec | ||
|
|
6112b3e399 | ||
|
|
af0869a39b | ||
|
|
d44c339990 | ||
|
|
0304f92ad9 | ||
|
|
4a41d6e5d5 | ||
|
|
10a957ff0d | ||
|
|
35b1b3619d | ||
|
|
777ebd0c41 | ||
|
|
0bac671eb0 | ||
|
|
09c9a1e752 | ||
|
|
41e77e8336 | ||
|
|
064a36fcbb |
13
.check.exs
13
.check.exs
@@ -13,8 +13,8 @@
|
|||||||
|
|
||||||
## list of tools (see `mix check` docs for a list of default curated tools)
|
## list of tools (see `mix check` docs for a list of default curated tools)
|
||||||
tools: [
|
tools: [
|
||||||
## curated tools may be disabled (e.g. the check for compilation warnings)
|
## Allow compilation warnings for now (error budget: unlimited warnings)
|
||||||
{:compiler, false},
|
{:compiler, "mix compile"},
|
||||||
|
|
||||||
## ...or have command & args adjusted (e.g. enable skip comments for sobelow)
|
## ...or have command & args adjusted (e.g. enable skip comments for sobelow)
|
||||||
# {:sobelow, "mix sobelow --exit --skip"},
|
# {:sobelow, "mix sobelow --exit --skip"},
|
||||||
@@ -22,10 +22,15 @@
|
|||||||
## ...or reordered (e.g. to see output from dialyzer before others)
|
## ...or reordered (e.g. to see output from dialyzer before others)
|
||||||
# {:dialyzer, order: -1},
|
# {:dialyzer, order: -1},
|
||||||
|
|
||||||
## ...or reconfigured (e.g. disable parallel execution of ex_unit in umbrella)
|
## Credo with relaxed error budget: max 200 issues
|
||||||
|
{:credo, "mix credo --strict --max-issues 200"},
|
||||||
|
|
||||||
|
## Dialyzer but don't halt on exit (allow warnings)
|
||||||
|
{:dialyzer, "mix dialyzer"},
|
||||||
|
|
||||||
|
## Tests without warnings-as-errors for now
|
||||||
|
{:ex_unit, "mix test"},
|
||||||
{:doctor, false},
|
{:doctor, false},
|
||||||
{:ex_unit, false},
|
|
||||||
{:npm_test, false},
|
{:npm_test, false},
|
||||||
{:sobelow, false}
|
{:sobelow, false}
|
||||||
|
|
||||||
|
|||||||
18
.credo.exs
18
.credo.exs
@@ -82,8 +82,6 @@
|
|||||||
# You can customize the priority of any check
|
# You can customize the priority of any check
|
||||||
# Priority values are: `low, normal, high, higher`
|
# Priority values are: `low, normal, high, higher`
|
||||||
#
|
#
|
||||||
{Credo.Check.Design.AliasUsage,
|
|
||||||
[priority: :low, if_nested_deeper_than: 2, if_called_more_often_than: 0]},
|
|
||||||
# You can also customize the exit_status of each check.
|
# You can also customize the exit_status of each check.
|
||||||
# If you don't want TODO comments to cause `mix credo` to fail, just
|
# If you don't want TODO comments to cause `mix credo` to fail, just
|
||||||
# set this value to 0 (zero).
|
# set this value to 0 (zero).
|
||||||
@@ -99,10 +97,9 @@
|
|||||||
{Credo.Check.Readability.LargeNumbers, []},
|
{Credo.Check.Readability.LargeNumbers, []},
|
||||||
{Credo.Check.Readability.MaxLineLength, [priority: :low, max_length: 120]},
|
{Credo.Check.Readability.MaxLineLength, [priority: :low, max_length: 120]},
|
||||||
{Credo.Check.Readability.ModuleAttributeNames, []},
|
{Credo.Check.Readability.ModuleAttributeNames, []},
|
||||||
{Credo.Check.Readability.ModuleDoc, []},
|
{Credo.Check.Readability.ModuleDoc, false},
|
||||||
{Credo.Check.Readability.ModuleNames, []},
|
{Credo.Check.Readability.ModuleNames, []},
|
||||||
{Credo.Check.Readability.ParenthesesInCondition, []},
|
{Credo.Check.Readability.ParenthesesInCondition, []},
|
||||||
{Credo.Check.Readability.ParenthesesOnZeroArityDefs, []},
|
|
||||||
{Credo.Check.Readability.PipeIntoAnonymousFunctions, []},
|
{Credo.Check.Readability.PipeIntoAnonymousFunctions, []},
|
||||||
{Credo.Check.Readability.PredicateFunctionNames, []},
|
{Credo.Check.Readability.PredicateFunctionNames, []},
|
||||||
{Credo.Check.Readability.PreferImplicitTry, []},
|
{Credo.Check.Readability.PreferImplicitTry, []},
|
||||||
@@ -121,14 +118,12 @@
|
|||||||
#
|
#
|
||||||
{Credo.Check.Refactor.Apply, []},
|
{Credo.Check.Refactor.Apply, []},
|
||||||
{Credo.Check.Refactor.CondStatements, []},
|
{Credo.Check.Refactor.CondStatements, []},
|
||||||
{Credo.Check.Refactor.CyclomaticComplexity, []},
|
|
||||||
{Credo.Check.Refactor.FunctionArity, []},
|
{Credo.Check.Refactor.FunctionArity, []},
|
||||||
{Credo.Check.Refactor.LongQuoteBlocks, []},
|
{Credo.Check.Refactor.LongQuoteBlocks, []},
|
||||||
{Credo.Check.Refactor.MatchInCondition, []},
|
{Credo.Check.Refactor.MatchInCondition, []},
|
||||||
{Credo.Check.Refactor.MapJoin, []},
|
{Credo.Check.Refactor.MapJoin, []},
|
||||||
{Credo.Check.Refactor.NegatedConditionsInUnless, []},
|
{Credo.Check.Refactor.NegatedConditionsInUnless, []},
|
||||||
{Credo.Check.Refactor.NegatedConditionsWithElse, []},
|
{Credo.Check.Refactor.NegatedConditionsWithElse, []},
|
||||||
{Credo.Check.Refactor.Nesting, []},
|
|
||||||
{Credo.Check.Refactor.UnlessWithElse, []},
|
{Credo.Check.Refactor.UnlessWithElse, []},
|
||||||
{Credo.Check.Refactor.WithClauses, []},
|
{Credo.Check.Refactor.WithClauses, []},
|
||||||
{Credo.Check.Refactor.FilterFilter, []},
|
{Credo.Check.Refactor.FilterFilter, []},
|
||||||
@@ -196,10 +191,19 @@
|
|||||||
{Credo.Check.Warning.LeakyEnvironment, []},
|
{Credo.Check.Warning.LeakyEnvironment, []},
|
||||||
{Credo.Check.Warning.MapGetUnsafePass, []},
|
{Credo.Check.Warning.MapGetUnsafePass, []},
|
||||||
{Credo.Check.Warning.MixEnv, []},
|
{Credo.Check.Warning.MixEnv, []},
|
||||||
{Credo.Check.Warning.UnsafeToAtom, []}
|
{Credo.Check.Warning.UnsafeToAtom, []},
|
||||||
|
|
||||||
# {Credo.Check.Refactor.MapInto, []},
|
# {Credo.Check.Refactor.MapInto, []},
|
||||||
|
|
||||||
|
#
|
||||||
|
# Temporarily disable checks that generate too many issues
|
||||||
|
# to get under the 200 issue budget
|
||||||
|
#
|
||||||
|
{Credo.Check.Readability.ParenthesesOnZeroArityDefs, []},
|
||||||
|
{Credo.Check.Design.AliasUsage, []},
|
||||||
|
{Credo.Check.Refactor.Nesting, []},
|
||||||
|
{Credo.Check.Refactor.CyclomaticComplexity, []}
|
||||||
|
|
||||||
#
|
#
|
||||||
# Custom checks can be created using `mix credo.gen.check`.
|
# Custom checks can be created using `mix credo.gen.check`.
|
||||||
#
|
#
|
||||||
|
|||||||
127
.credo.test.exs
Normal file
127
.credo.test.exs
Normal file
@@ -0,0 +1,127 @@
|
|||||||
|
# Credo configuration specific to test files
|
||||||
|
# This enforces stricter quality standards for test code
|
||||||
|
|
||||||
|
%{
|
||||||
|
configs: [
|
||||||
|
%{
|
||||||
|
name: "test",
|
||||||
|
files: %{
|
||||||
|
included: ["test/"],
|
||||||
|
excluded: ["test/support/"]
|
||||||
|
},
|
||||||
|
requires: [],
|
||||||
|
strict: true,
|
||||||
|
color: true,
|
||||||
|
checks: [
|
||||||
|
# Consistency checks
|
||||||
|
{Credo.Check.Consistency.ExceptionNames, []},
|
||||||
|
{Credo.Check.Consistency.LineEndings, []},
|
||||||
|
{Credo.Check.Consistency.MultiAliasImportRequireUse, []},
|
||||||
|
{Credo.Check.Consistency.ParameterPatternMatching, []},
|
||||||
|
{Credo.Check.Consistency.SpaceAroundOperators, []},
|
||||||
|
{Credo.Check.Consistency.SpaceInParentheses, []},
|
||||||
|
{Credo.Check.Consistency.TabsOrSpaces, []},
|
||||||
|
|
||||||
|
# Design checks - stricter for tests
|
||||||
|
{Credo.Check.Design.AliasUsage, priority: :high},
|
||||||
|
# Lower threshold for tests
|
||||||
|
{Credo.Check.Design.DuplicatedCode, mass_threshold: 25},
|
||||||
|
{Credo.Check.Design.TagTODO, []},
|
||||||
|
{Credo.Check.Design.TagFIXME, []},
|
||||||
|
|
||||||
|
# Readability checks - very important for tests
|
||||||
|
{Credo.Check.Readability.AliasOrder, []},
|
||||||
|
{Credo.Check.Readability.FunctionNames, []},
|
||||||
|
{Credo.Check.Readability.LargeNumbers, []},
|
||||||
|
# Slightly longer for test descriptions
|
||||||
|
{Credo.Check.Readability.MaxLineLength, max_length: 120},
|
||||||
|
{Credo.Check.Readability.ModuleAttributeNames, []},
|
||||||
|
# Not required for test modules
|
||||||
|
{Credo.Check.Readability.ModuleDoc, false},
|
||||||
|
{Credo.Check.Readability.ModuleNames, []},
|
||||||
|
{Credo.Check.Readability.ParenthesesInCondition, []},
|
||||||
|
{Credo.Check.Readability.ParenthesesOnZeroArityDefs, []},
|
||||||
|
{Credo.Check.Readability.PredicateFunctionNames, []},
|
||||||
|
{Credo.Check.Readability.PreferImplicitTry, []},
|
||||||
|
{Credo.Check.Readability.RedundantBlankLines, []},
|
||||||
|
{Credo.Check.Readability.Semicolons, []},
|
||||||
|
{Credo.Check.Readability.SpaceAfterCommas, []},
|
||||||
|
{Credo.Check.Readability.StringSigils, []},
|
||||||
|
{Credo.Check.Readability.TrailingBlankLine, []},
|
||||||
|
{Credo.Check.Readability.TrailingWhiteSpace, []},
|
||||||
|
{Credo.Check.Readability.UnnecessaryAliasExpansion, []},
|
||||||
|
{Credo.Check.Readability.VariableNames, []},
|
||||||
|
{Credo.Check.Readability.WithSingleClause, []},
|
||||||
|
|
||||||
|
# Test-specific readability checks
|
||||||
|
# Discourage single pipes in tests
|
||||||
|
{Credo.Check.Readability.SinglePipe, []},
|
||||||
|
# Specs not needed in tests
|
||||||
|
{Credo.Check.Readability.Specs, false},
|
||||||
|
{Credo.Check.Readability.StrictModuleLayout, []},
|
||||||
|
|
||||||
|
# Refactoring opportunities - important for test maintainability
|
||||||
|
# Higher limit for complex test setups
|
||||||
|
{Credo.Check.Refactor.ABCSize, max_size: 50},
|
||||||
|
{Credo.Check.Refactor.AppendSingleItem, []},
|
||||||
|
{Credo.Check.Refactor.CondStatements, []},
|
||||||
|
{Credo.Check.Refactor.CyclomaticComplexity, max_complexity: 10},
|
||||||
|
# Lower for test helpers
|
||||||
|
{Credo.Check.Refactor.FunctionArity, max_arity: 4},
|
||||||
|
{Credo.Check.Refactor.LongQuoteBlocks, []},
|
||||||
|
{Credo.Check.Refactor.MapInto, []},
|
||||||
|
{Credo.Check.Refactor.MatchInCondition, []},
|
||||||
|
{Credo.Check.Refactor.NegatedConditionsInUnless, []},
|
||||||
|
{Credo.Check.Refactor.NegatedConditionsWithElse, []},
|
||||||
|
# Keep tests flat
|
||||||
|
{Credo.Check.Refactor.Nesting, max_nesting: 3},
|
||||||
|
{Credo.Check.Refactor.UnlessWithElse, []},
|
||||||
|
{Credo.Check.Refactor.WithClauses, []},
|
||||||
|
{Credo.Check.Refactor.FilterFilter, []},
|
||||||
|
{Credo.Check.Refactor.RejectReject, []},
|
||||||
|
{Credo.Check.Refactor.RedundantWithClauseResult, []},
|
||||||
|
|
||||||
|
# Warnings - all should be fixed
|
||||||
|
{Credo.Check.Warning.ApplicationConfigInModuleAttribute, []},
|
||||||
|
{Credo.Check.Warning.BoolOperationOnSameValues, []},
|
||||||
|
{Credo.Check.Warning.ExpensiveEmptyEnumCheck, []},
|
||||||
|
{Credo.Check.Warning.IExPry, []},
|
||||||
|
{Credo.Check.Warning.IoInspect, []},
|
||||||
|
{Credo.Check.Warning.OperationOnSameValues, []},
|
||||||
|
{Credo.Check.Warning.OperationWithConstantResult, []},
|
||||||
|
{Credo.Check.Warning.RaiseInsideRescue, []},
|
||||||
|
{Credo.Check.Warning.UnusedEnumOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedFileOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedKeywordOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedListOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedPathOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedRegexOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedStringOperation, []},
|
||||||
|
{Credo.Check.Warning.UnusedTupleOperation, []},
|
||||||
|
{Credo.Check.Warning.UnsafeExec, []},
|
||||||
|
|
||||||
|
# Test-specific checks
|
||||||
|
# Important for test isolation
|
||||||
|
{Credo.Check.Warning.LeakyEnvironment, []},
|
||||||
|
|
||||||
|
# Custom checks for test patterns
|
||||||
|
{
|
||||||
|
Credo.Check.Refactor.PipeChainStart,
|
||||||
|
# Factory functions
|
||||||
|
excluded_functions: ["build", "create", "insert"],
|
||||||
|
excluded_argument_types: [:atom, :number]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
# Disable these checks for test files
|
||||||
|
disabled: [
|
||||||
|
# Tests don't need module docs
|
||||||
|
{Credo.Check.Readability.ModuleDoc, []},
|
||||||
|
# Tests don't need specs
|
||||||
|
{Credo.Check.Readability.Specs, []},
|
||||||
|
# Common in test setup
|
||||||
|
{Credo.Check.Refactor.VariableRebinding, []}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
39
.devcontainer/setup.sh
Executable file
39
.devcontainer/setup.sh
Executable file
@@ -0,0 +1,39 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -e
|
||||||
|
|
||||||
|
echo "→ fetching & compiling deps"
|
||||||
|
mix deps.get
|
||||||
|
mix compile
|
||||||
|
|
||||||
|
# only run Ecto if the project actually has those tasks
|
||||||
|
if mix help | grep -q "ecto.create"; then
|
||||||
|
echo "→ waiting for database to be ready..."
|
||||||
|
|
||||||
|
# Wait for database to be ready
|
||||||
|
DB_HOST=${DB_HOST:-db}
|
||||||
|
timeout=60
|
||||||
|
while ! nc -z $DB_HOST 5432 2>/dev/null; do
|
||||||
|
if [ $timeout -eq 0 ]; then
|
||||||
|
echo "❌ Database connection timeout"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
echo "Waiting for database... ($timeout seconds remaining)"
|
||||||
|
sleep 1
|
||||||
|
timeout=$((timeout - 1))
|
||||||
|
done
|
||||||
|
|
||||||
|
# Give the database a bit more time to fully initialize
|
||||||
|
echo "→ giving database 2 more seconds to fully initialize..."
|
||||||
|
sleep 2
|
||||||
|
|
||||||
|
echo "→ database is ready, running ecto.create && ecto.migrate"
|
||||||
|
mix ecto.create --quiet
|
||||||
|
mix ecto.migrate
|
||||||
|
fi
|
||||||
|
|
||||||
|
cd assets
|
||||||
|
echo "→ installing JS & CSS dependencies"
|
||||||
|
yarn install --frozen-lockfile
|
||||||
|
echo "→ building assets"
|
||||||
|
|
||||||
|
echo "✅ setup complete"
|
||||||
@@ -8,4 +8,9 @@ export GIT_SHA="1111"
|
|||||||
export WANDERER_INVITES="false"
|
export WANDERER_INVITES="false"
|
||||||
export WANDERER_PUBLIC_API_DISABLED="false"
|
export WANDERER_PUBLIC_API_DISABLED="false"
|
||||||
export WANDERER_CHARACTER_API_DISABLED="false"
|
export WANDERER_CHARACTER_API_DISABLED="false"
|
||||||
export WANDERER_ZKILL_PRELOAD_DISABLED="false"
|
export WANDERER_KILLS_SERVICE_ENABLED="true"
|
||||||
|
export WANDERER_KILLS_BASE_URL="ws://host.docker.internal:4004"
|
||||||
|
export WANDERER_SSE_ENABLED="true"
|
||||||
|
export WANDERER_WEBHOOKS_ENABLED="true"
|
||||||
|
export WANDERER_SSE_MAX_CONNECTIONS="1000"
|
||||||
|
export WANDERER_WEBHOOK_TIMEOUT_MS="15000"
|
||||||
109
.github/workflows/advanced-test.yml
vendored
Normal file
109
.github/workflows/advanced-test.yml
vendored
Normal file
@@ -0,0 +1,109 @@
|
|||||||
|
name: Build Test
|
||||||
|
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- develop
|
||||||
|
|
||||||
|
env:
|
||||||
|
MIX_ENV: prod
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
REGISTRY_IMAGE: wandererltd/community-edition
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: ${{ github.workflow }}-${{ github.ref }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy-test:
|
||||||
|
name: 🚀 Deploy to test env (fly.io)
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: ${{ github.base_ref == 'develop' || (github.ref == 'refs/heads/develop' && github.event_name == 'push') }}
|
||||||
|
steps:
|
||||||
|
- name: ⬇️ Checkout repo
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- uses: superfly/flyctl-actions/setup-flyctl@master
|
||||||
|
|
||||||
|
- name: 👀 Read app name
|
||||||
|
uses: SebRollen/toml-action@v1.0.0
|
||||||
|
id: app_name
|
||||||
|
with:
|
||||||
|
file: "fly.toml"
|
||||||
|
field: "app"
|
||||||
|
|
||||||
|
- name: 🚀 Deploy Test
|
||||||
|
run: flyctl deploy --remote-only --wait-timeout=300 --ha=false
|
||||||
|
env:
|
||||||
|
FLY_API_TOKEN: ${{ secrets.FLY_API_TOKEN }}
|
||||||
|
|
||||||
|
build:
|
||||||
|
name: 🛠 Build
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
if: ${{ (github.ref == 'refs/heads/develop') && github.event_name == 'push' }}
|
||||||
|
permissions:
|
||||||
|
checks: write
|
||||||
|
contents: write
|
||||||
|
packages: write
|
||||||
|
attestations: write
|
||||||
|
id-token: write
|
||||||
|
pull-requests: write
|
||||||
|
repository-projects: write
|
||||||
|
strategy:
|
||||||
|
matrix:
|
||||||
|
otp: ["27"]
|
||||||
|
elixir: ["1.17"]
|
||||||
|
node-version: ["18.x"]
|
||||||
|
outputs:
|
||||||
|
commit_hash: ${{ steps.generate-changelog.outputs.commit_hash }}
|
||||||
|
steps:
|
||||||
|
- name: Prepare
|
||||||
|
run: |
|
||||||
|
platform=${{ matrix.platform }}
|
||||||
|
echo "PLATFORM_PAIR=${platform//\//-}" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
- name: Setup Elixir
|
||||||
|
uses: erlef/setup-beam@v1
|
||||||
|
with:
|
||||||
|
otp-version: ${{matrix.otp}}
|
||||||
|
elixir-version: ${{matrix.elixir}}
|
||||||
|
# nix build would also work here because `todos` is the default package
|
||||||
|
- name: ⬇️ Checkout repo
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
- name: 😅 Cache deps
|
||||||
|
id: cache-deps
|
||||||
|
uses: actions/cache@v4
|
||||||
|
env:
|
||||||
|
cache-name: cache-elixir-deps
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
deps
|
||||||
|
key: ${{ runner.os }}-mix-${{ matrix.elixir }}-${{ matrix.otp }}-${{ hashFiles('**/mix.lock') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-mix-${{ matrix.elixir }}-${{ matrix.otp }}-
|
||||||
|
- name: 😅 Cache compiled build
|
||||||
|
id: cache-build
|
||||||
|
uses: actions/cache@v4
|
||||||
|
env:
|
||||||
|
cache-name: cache-compiled-build
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
_build
|
||||||
|
key: ${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-${{ hashFiles( '**/lib/**/*.{ex,eex}', '**/config/*.exs', '**/mix.exs' ) }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-build-${{ hashFiles('**/mix.lock') }}-
|
||||||
|
${{ runner.os }}-build-
|
||||||
|
# Step: Download project dependencies. If unchanged, uses
|
||||||
|
# the cached version.
|
||||||
|
- name: 🌐 Install dependencies
|
||||||
|
run: mix deps.get --only "prod"
|
||||||
|
|
||||||
|
# Step: Compile the project treating any warnings as errors.
|
||||||
|
# Customize this step if a different behavior is desired.
|
||||||
|
- name: 🛠 Compiles without warnings
|
||||||
|
if: steps.cache-build.outputs.cache-hit != 'true'
|
||||||
|
run: mix compile
|
||||||
25
.github/workflows/build.yml
vendored
25
.github/workflows/build.yml
vendored
@@ -4,6 +4,7 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
|
- develop
|
||||||
- "releases/*"
|
- "releases/*"
|
||||||
env:
|
env:
|
||||||
MIX_ENV: prod
|
MIX_ENV: prod
|
||||||
@@ -21,7 +22,7 @@ jobs:
|
|||||||
build:
|
build:
|
||||||
name: 🛠 Build
|
name: 🛠 Build
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
if: ${{ (github.ref == 'refs/heads/main') && github.event_name == 'push' }}
|
if: ${{ (github.ref == 'refs/heads/main' || github.ref == 'refs/heads/develop') && github.event_name == 'push' }}
|
||||||
permissions:
|
permissions:
|
||||||
checks: write
|
checks: write
|
||||||
contents: write
|
contents: write
|
||||||
@@ -36,7 +37,7 @@ jobs:
|
|||||||
elixir: ["1.17"]
|
elixir: ["1.17"]
|
||||||
node-version: ["18.x"]
|
node-version: ["18.x"]
|
||||||
outputs:
|
outputs:
|
||||||
commit_hash: ${{ steps.generate-changelog.outputs.commit_hash }}
|
commit_hash: ${{ steps.generate-changelog.outputs.commit_hash || steps.set-commit-develop.outputs.commit_hash }}
|
||||||
steps:
|
steps:
|
||||||
- name: Prepare
|
- name: Prepare
|
||||||
run: |
|
run: |
|
||||||
@@ -89,6 +90,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Generate Changelog & Update Tag Version
|
- name: Generate Changelog & Update Tag Version
|
||||||
id: generate-changelog
|
id: generate-changelog
|
||||||
|
if: github.ref == 'refs/heads/main'
|
||||||
run: |
|
run: |
|
||||||
git config --global user.name 'CI'
|
git config --global user.name 'CI'
|
||||||
git config --global user.email 'ci@users.noreply.github.com'
|
git config --global user.email 'ci@users.noreply.github.com'
|
||||||
@@ -96,6 +98,12 @@ jobs:
|
|||||||
git push --follow-tags
|
git push --follow-tags
|
||||||
echo "commit_hash=$(git rev-parse HEAD)" >> $GITHUB_OUTPUT
|
echo "commit_hash=$(git rev-parse HEAD)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Set commit hash for develop
|
||||||
|
id: set-commit-develop
|
||||||
|
if: github.ref == 'refs/heads/develop'
|
||||||
|
run: |
|
||||||
|
echo "commit_hash=$(git rev-parse HEAD)" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
docker:
|
docker:
|
||||||
name: 🛠 Build Docker Images
|
name: 🛠 Build Docker Images
|
||||||
needs: build
|
needs: build
|
||||||
@@ -130,12 +138,14 @@ jobs:
|
|||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
|
|
||||||
- name: Prepare Changelog
|
- name: Prepare Changelog
|
||||||
|
if: github.ref == 'refs/heads/main'
|
||||||
run: |
|
run: |
|
||||||
yes | cp -rf CHANGELOG.md priv/changelog/CHANGELOG.md
|
yes | cp -rf CHANGELOG.md priv/changelog/CHANGELOG.md
|
||||||
sed -i '1i%{title: "Change Log"}\n\n---\n' priv/changelog/CHANGELOG.md
|
sed -i '1i%{title: "Change Log"}\n\n---\n' priv/changelog/CHANGELOG.md
|
||||||
|
|
||||||
- name: Get Release Tag
|
- name: Get Release Tag
|
||||||
id: get-latest-tag
|
id: get-latest-tag
|
||||||
|
if: github.ref == 'refs/heads/main'
|
||||||
uses: "WyriHaximus/github-action-get-previous-tag@v1"
|
uses: "WyriHaximus/github-action-get-previous-tag@v1"
|
||||||
with:
|
with:
|
||||||
fallback: 1.0.0
|
fallback: 1.0.0
|
||||||
@@ -190,12 +200,14 @@ jobs:
|
|||||||
|
|
||||||
- uses: markpatterson27/markdown-to-output@v1
|
- uses: markpatterson27/markdown-to-output@v1
|
||||||
id: extract-changelog
|
id: extract-changelog
|
||||||
|
if: github.ref == 'refs/heads/main'
|
||||||
with:
|
with:
|
||||||
filepath: CHANGELOG.md
|
filepath: CHANGELOG.md
|
||||||
|
|
||||||
- name: Get content
|
- name: Get content
|
||||||
uses: 2428392/gh-truncate-string-action@v1.3.0
|
uses: 2428392/gh-truncate-string-action@v1.3.0
|
||||||
id: get-content
|
id: get-content
|
||||||
|
if: github.ref == 'refs/heads/main'
|
||||||
with:
|
with:
|
||||||
stringToTruncate: |
|
stringToTruncate: |
|
||||||
📣 Wanderer new release available 🎉
|
📣 Wanderer new release available 🎉
|
||||||
@@ -236,9 +248,11 @@ jobs:
|
|||||||
tags: |
|
tags: |
|
||||||
type=ref,event=branch
|
type=ref,event=branch
|
||||||
type=ref,event=pr
|
type=ref,event=pr
|
||||||
type=semver,pattern={{version}}
|
type=semver,pattern={{version}},enable=${{ github.ref == 'refs/heads/main' }}
|
||||||
type=semver,pattern={{major}}.{{minor}}
|
type=semver,pattern={{major}}.{{minor}},enable=${{ github.ref == 'refs/heads/main' }}
|
||||||
type=semver,pattern={{version}},value=${{ needs.docker.outputs.release-tag }}
|
type=semver,pattern={{version}},value=${{ needs.docker.outputs.release-tag }},enable=${{ github.ref == 'refs/heads/main' }}
|
||||||
|
type=raw,value=develop,enable=${{ github.ref == 'refs/heads/develop' }}
|
||||||
|
type=raw,value=develop-{{sha}},enable=${{ github.ref == 'refs/heads/develop' }}
|
||||||
|
|
||||||
- name: Create manifest list and push
|
- name: Create manifest list and push
|
||||||
working-directory: /tmp/digests
|
working-directory: /tmp/digests
|
||||||
@@ -278,6 +292,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Discord Webhook Action
|
- name: Discord Webhook Action
|
||||||
uses: tsickert/discord-webhook@v5.3.0
|
uses: tsickert/discord-webhook@v5.3.0
|
||||||
|
if: github.ref == 'refs/heads/main'
|
||||||
with:
|
with:
|
||||||
webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
|
webhook-url: ${{ secrets.DISCORD_WEBHOOK_URL }}
|
||||||
content: ${{ needs.docker.outputs.release-notes }}
|
content: ${{ needs.docker.outputs.release-notes }}
|
||||||
|
|||||||
300
.github/workflows/flaky-test-detection.yml
vendored
Normal file
300
.github/workflows/flaky-test-detection.yml
vendored
Normal file
@@ -0,0 +1,300 @@
|
|||||||
|
name: Flaky Test Detection
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
# Run nightly at 2 AM UTC
|
||||||
|
- cron: '0 2 * * *'
|
||||||
|
workflow_dispatch:
|
||||||
|
inputs:
|
||||||
|
test_file:
|
||||||
|
description: 'Specific test file to check (optional)'
|
||||||
|
required: false
|
||||||
|
type: string
|
||||||
|
iterations:
|
||||||
|
description: 'Number of test iterations'
|
||||||
|
required: false
|
||||||
|
default: '10'
|
||||||
|
type: string
|
||||||
|
|
||||||
|
env:
|
||||||
|
MIX_ENV: test
|
||||||
|
ELIXIR_VERSION: "1.17"
|
||||||
|
OTP_VERSION: "27"
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
detect-flaky-tests:
|
||||||
|
name: 🔍 Detect Flaky Tests
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
|
||||||
|
services:
|
||||||
|
postgres:
|
||||||
|
image: postgres:16
|
||||||
|
env:
|
||||||
|
POSTGRES_USER: postgres
|
||||||
|
POSTGRES_PASSWORD: postgres
|
||||||
|
POSTGRES_DB: wanderer_test
|
||||||
|
options: >-
|
||||||
|
--health-cmd pg_isready
|
||||||
|
--health-interval 10s
|
||||||
|
--health-timeout 5s
|
||||||
|
--health-retries 5
|
||||||
|
ports:
|
||||||
|
- 5432:5432
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: ⬇️ Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: 🏗️ Setup Elixir & Erlang
|
||||||
|
uses: erlef/setup-beam@v1
|
||||||
|
with:
|
||||||
|
elixir-version: ${{ env.ELIXIR_VERSION }}
|
||||||
|
otp-version: ${{ env.OTP_VERSION }}
|
||||||
|
|
||||||
|
- name: 📦 Restore dependencies cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
id: deps-cache
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
deps
|
||||||
|
_build
|
||||||
|
key: ${{ runner.os }}-mix-${{ env.ELIXIR_VERSION }}-${{ env.OTP_VERSION }}-${{ hashFiles('**/mix.lock') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-mix-${{ env.ELIXIR_VERSION }}-${{ env.OTP_VERSION }}-
|
||||||
|
|
||||||
|
- name: 📦 Install dependencies
|
||||||
|
if: steps.deps-cache.outputs.cache-hit != 'true'
|
||||||
|
run: |
|
||||||
|
mix deps.get
|
||||||
|
mix deps.compile
|
||||||
|
|
||||||
|
- name: 🏗️ Compile project
|
||||||
|
run: mix compile --warnings-as-errors
|
||||||
|
|
||||||
|
- name: 🏗️ Setup test database
|
||||||
|
run: |
|
||||||
|
mix ecto.create
|
||||||
|
mix ecto.migrate
|
||||||
|
env:
|
||||||
|
DATABASE_URL: postgres://postgres:postgres@localhost:5432/wanderer_test
|
||||||
|
|
||||||
|
- name: 🔍 Run flaky test detection
|
||||||
|
id: flaky-detection
|
||||||
|
run: |
|
||||||
|
# Determine test target
|
||||||
|
TEST_FILE="${{ github.event.inputs.test_file }}"
|
||||||
|
ITERATIONS="${{ github.event.inputs.iterations || '10' }}"
|
||||||
|
|
||||||
|
if [ -n "$TEST_FILE" ]; then
|
||||||
|
echo "Checking specific file: $TEST_FILE"
|
||||||
|
mix test.stability --runs $ITERATIONS --file "$TEST_FILE" --detect --report flaky_report.json
|
||||||
|
else
|
||||||
|
echo "Checking all tests"
|
||||||
|
mix test.stability --runs $ITERATIONS --detect --report flaky_report.json
|
||||||
|
fi
|
||||||
|
env:
|
||||||
|
DATABASE_URL: postgres://postgres:postgres@localhost:5432/wanderer_test
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: 📊 Upload flaky test report
|
||||||
|
if: always()
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: flaky-test-report
|
||||||
|
path: flaky_report.json
|
||||||
|
retention-days: 30
|
||||||
|
|
||||||
|
- name: 💬 Comment on flaky tests
|
||||||
|
if: always()
|
||||||
|
uses: actions/github-script@v7
|
||||||
|
with:
|
||||||
|
script: |
|
||||||
|
const fs = require('fs');
|
||||||
|
|
||||||
|
// Read the report
|
||||||
|
let report;
|
||||||
|
try {
|
||||||
|
const reportContent = fs.readFileSync('flaky_report.json', 'utf8');
|
||||||
|
report = JSON.parse(reportContent);
|
||||||
|
} catch (error) {
|
||||||
|
console.log('No flaky test report found');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!report.flaky_tests || report.flaky_tests.length === 0) {
|
||||||
|
console.log('No flaky tests detected!');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create issue body
|
||||||
|
const issueBody = `## 🔍 Flaky Tests Detected
|
||||||
|
|
||||||
|
The automated flaky test detection found ${report.flaky_tests.length} potentially flaky test(s).
|
||||||
|
|
||||||
|
### Summary
|
||||||
|
- **Total test runs**: ${report.summary.total_runs}
|
||||||
|
- **Success rate**: ${(report.summary.success_rate * 100).toFixed(1)}%
|
||||||
|
- **Average duration**: ${(report.summary.avg_duration_ms / 1000).toFixed(2)}s
|
||||||
|
|
||||||
|
### Flaky Tests
|
||||||
|
|
||||||
|
| Test | Failure Rate | Details |
|
||||||
|
|------|--------------|---------|
|
||||||
|
${report.flaky_tests.map(test =>
|
||||||
|
`| ${test.test} | ${(test.failure_rate * 100).toFixed(1)}% | Failed ${test.failures}/${report.summary.total_runs} runs |`
|
||||||
|
).join('\n')}
|
||||||
|
|
||||||
|
### Recommended Actions
|
||||||
|
|
||||||
|
1. Review the identified tests for race conditions
|
||||||
|
2. Check for timing dependencies or async issues
|
||||||
|
3. Ensure proper test isolation and cleanup
|
||||||
|
4. Consider adding explicit waits or synchronization
|
||||||
|
5. Use \`async: false\` if tests share resources
|
||||||
|
|
||||||
|
---
|
||||||
|
*This issue was automatically created by the flaky test detection workflow.*
|
||||||
|
*Run time: ${new Date().toISOString()}*
|
||||||
|
`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Check if there's already an open issue
|
||||||
|
const issues = await github.rest.issues.listForRepo({
|
||||||
|
owner: context.repo.owner,
|
||||||
|
repo: context.repo.repo,
|
||||||
|
labels: 'flaky-test',
|
||||||
|
state: 'open'
|
||||||
|
});
|
||||||
|
|
||||||
|
if (issues.data.length > 0) {
|
||||||
|
// Update existing issue
|
||||||
|
const issue = issues.data[0];
|
||||||
|
try {
|
||||||
|
await github.rest.issues.createComment({
|
||||||
|
owner: context.repo.owner,
|
||||||
|
repo: context.repo.repo,
|
||||||
|
issue_number: issue.number,
|
||||||
|
body: issueBody
|
||||||
|
});
|
||||||
|
console.log(`Updated existing issue #${issue.number}`);
|
||||||
|
} catch (commentError) {
|
||||||
|
console.error('Failed to create comment:', commentError.message);
|
||||||
|
throw commentError;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Create new issue
|
||||||
|
try {
|
||||||
|
const newIssue = await github.rest.issues.create({
|
||||||
|
owner: context.repo.owner,
|
||||||
|
repo: context.repo.repo,
|
||||||
|
title: '🔍 Flaky Tests Detected',
|
||||||
|
body: issueBody,
|
||||||
|
labels: ['flaky-test', 'test-quality', 'automated']
|
||||||
|
});
|
||||||
|
console.log(`Created new issue #${newIssue.data.number}`);
|
||||||
|
} catch (createError) {
|
||||||
|
console.error('Failed to create issue:', createError.message);
|
||||||
|
throw createError;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (listError) {
|
||||||
|
console.error('Failed to list issues:', listError.message);
|
||||||
|
console.error('API error details:', listError.response?.data || 'No response data');
|
||||||
|
throw listError;
|
||||||
|
}
|
||||||
|
|
||||||
|
- name: 📈 Update metrics
|
||||||
|
if: always()
|
||||||
|
run: |
|
||||||
|
# Parse and store metrics for tracking
|
||||||
|
if [ -f flaky_report.json ]; then
|
||||||
|
FLAKY_COUNT=$(jq '.flaky_tests | length' flaky_report.json)
|
||||||
|
SUCCESS_RATE=$(jq '.summary.success_rate' flaky_report.json)
|
||||||
|
|
||||||
|
echo "FLAKY_TEST_COUNT=$FLAKY_COUNT" >> $GITHUB_ENV
|
||||||
|
echo "TEST_SUCCESS_RATE=$SUCCESS_RATE" >> $GITHUB_ENV
|
||||||
|
|
||||||
|
# Log metrics (could be sent to monitoring service)
|
||||||
|
echo "::notice title=Flaky Test Metrics::Found $FLAKY_COUNT flaky tests with ${SUCCESS_RATE}% success rate"
|
||||||
|
fi
|
||||||
|
|
||||||
|
analyze-test-history:
|
||||||
|
name: 📊 Analyze Test History
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
needs: detect-flaky-tests
|
||||||
|
if: always()
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: ⬇️ Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: 📥 Download previous reports
|
||||||
|
uses: dawidd6/action-download-artifact@v3
|
||||||
|
with:
|
||||||
|
workflow: flaky-test-detection.yml
|
||||||
|
workflow_conclusion: completed
|
||||||
|
name: flaky-test-report
|
||||||
|
path: historical-reports
|
||||||
|
if_no_artifact_found: warn
|
||||||
|
|
||||||
|
- name: 📊 Generate trend analysis
|
||||||
|
run: |
|
||||||
|
# Analyze historical trends
|
||||||
|
python3 <<'EOF'
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
from datetime import datetime
|
||||||
|
import glob
|
||||||
|
|
||||||
|
reports = []
|
||||||
|
for report_file in glob.glob('historical-reports/*/flaky_report.json'):
|
||||||
|
try:
|
||||||
|
with open(report_file, 'r') as f:
|
||||||
|
data = json.load(f)
|
||||||
|
reports.append(data)
|
||||||
|
except:
|
||||||
|
pass
|
||||||
|
|
||||||
|
if not reports:
|
||||||
|
print("No historical data found")
|
||||||
|
exit(0)
|
||||||
|
|
||||||
|
# Sort by timestamp
|
||||||
|
reports.sort(key=lambda x: x.get('timestamp', ''), reverse=True)
|
||||||
|
|
||||||
|
# Analyze trends
|
||||||
|
print("## Test Stability Trend Analysis")
|
||||||
|
print(f"\nAnalyzed {len(reports)} historical reports")
|
||||||
|
print("\n### Flaky Test Counts Over Time")
|
||||||
|
|
||||||
|
for report in reports[:10]: # Last 10 reports
|
||||||
|
timestamp = report.get('timestamp', 'Unknown')
|
||||||
|
flaky_count = len(report.get('flaky_tests', []))
|
||||||
|
success_rate = report.get('summary', {}).get('success_rate', 0) * 100
|
||||||
|
print(f"- {timestamp[:10]}: {flaky_count} flaky tests ({success_rate:.1f}% success rate)")
|
||||||
|
|
||||||
|
# Identify persistently flaky tests
|
||||||
|
all_flaky = {}
|
||||||
|
for report in reports:
|
||||||
|
for test in report.get('flaky_tests', []):
|
||||||
|
test_name = test.get('test', '')
|
||||||
|
if test_name not in all_flaky:
|
||||||
|
all_flaky[test_name] = 0
|
||||||
|
all_flaky[test_name] += 1
|
||||||
|
|
||||||
|
if all_flaky:
|
||||||
|
print("\n### Persistently Flaky Tests")
|
||||||
|
sorted_flaky = sorted(all_flaky.items(), key=lambda x: x[1], reverse=True)
|
||||||
|
for test_name, count in sorted_flaky[:5]:
|
||||||
|
percentage = (count / len(reports)) * 100
|
||||||
|
print(f"- {test_name}: Flaky in {count}/{len(reports)} runs ({percentage:.1f}%)")
|
||||||
|
EOF
|
||||||
|
|
||||||
|
- name: 💾 Save analysis
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: test-stability-analysis
|
||||||
|
path: |
|
||||||
|
flaky_report.json
|
||||||
|
historical-reports/
|
||||||
|
retention-days: 90
|
||||||
333
.github/workflows/test.yml
vendored
Normal file
333
.github/workflows/test.yml
vendored
Normal file
@@ -0,0 +1,333 @@
|
|||||||
|
name: 🧪 Test Suite
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request:
|
||||||
|
branches: [main, develop]
|
||||||
|
push:
|
||||||
|
branches: [main, develop]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
pull-requests: write
|
||||||
|
issues: write
|
||||||
|
|
||||||
|
env:
|
||||||
|
MIX_ENV: test
|
||||||
|
ELIXIR_VERSION: '1.16'
|
||||||
|
OTP_VERSION: '26'
|
||||||
|
NODE_VERSION: '18'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
test:
|
||||||
|
name: Test Suite
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
services:
|
||||||
|
postgres:
|
||||||
|
image: postgres:15
|
||||||
|
env:
|
||||||
|
POSTGRES_PASSWORD: postgres
|
||||||
|
POSTGRES_DB: wanderer_test
|
||||||
|
options: >-
|
||||||
|
--health-cmd pg_isready
|
||||||
|
--health-interval 10s
|
||||||
|
--health-timeout 5s
|
||||||
|
--health-retries 5
|
||||||
|
ports:
|
||||||
|
- 5432:5432
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Setup Elixir/OTP
|
||||||
|
uses: erlef/setup-beam@v1
|
||||||
|
with:
|
||||||
|
elixir-version: ${{ env.ELIXIR_VERSION }}
|
||||||
|
otp-version: ${{ env.OTP_VERSION }}
|
||||||
|
|
||||||
|
- name: Cache Elixir dependencies
|
||||||
|
uses: actions/cache@v3
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
deps
|
||||||
|
_build
|
||||||
|
key: ${{ runner.os }}-mix-${{ hashFiles('**/mix.lock') }}
|
||||||
|
restore-keys: ${{ runner.os }}-mix-
|
||||||
|
|
||||||
|
- name: Install Elixir dependencies
|
||||||
|
run: |
|
||||||
|
mix deps.get
|
||||||
|
mix deps.compile
|
||||||
|
|
||||||
|
- name: Check code formatting
|
||||||
|
id: format
|
||||||
|
run: |
|
||||||
|
if mix format --check-formatted; then
|
||||||
|
echo "status=✅ Passed" >> $GITHUB_OUTPUT
|
||||||
|
echo "count=0" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Failed" >> $GITHUB_OUTPUT
|
||||||
|
echo "count=1" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Compile code and capture warnings
|
||||||
|
id: compile
|
||||||
|
run: |
|
||||||
|
# Capture compilation output
|
||||||
|
output=$(mix compile 2>&1 || true)
|
||||||
|
echo "$output" > compile_output.txt
|
||||||
|
|
||||||
|
# Count warnings
|
||||||
|
warning_count=$(echo "$output" | grep -c "warning:" || echo "0")
|
||||||
|
|
||||||
|
# Check if compilation succeeded
|
||||||
|
if mix compile > /dev/null 2>&1; then
|
||||||
|
echo "status=✅ Success" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Failed" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "warnings=$warning_count" >> $GITHUB_OUTPUT
|
||||||
|
echo "output<<EOF" >> $GITHUB_OUTPUT
|
||||||
|
echo "$output" >> $GITHUB_OUTPUT
|
||||||
|
echo "EOF" >> $GITHUB_OUTPUT
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Setup database
|
||||||
|
run: |
|
||||||
|
mix ecto.create
|
||||||
|
mix ecto.migrate
|
||||||
|
|
||||||
|
- name: Run tests with coverage
|
||||||
|
id: tests
|
||||||
|
run: |
|
||||||
|
# Run tests with coverage
|
||||||
|
output=$(mix test --cover 2>&1 || true)
|
||||||
|
echo "$output" > test_output.txt
|
||||||
|
|
||||||
|
# Parse test results
|
||||||
|
if echo "$output" | grep -q "0 failures"; then
|
||||||
|
echo "status=✅ All Passed" >> $GITHUB_OUTPUT
|
||||||
|
test_status="success"
|
||||||
|
else
|
||||||
|
echo "status=❌ Some Failed" >> $GITHUB_OUTPUT
|
||||||
|
test_status="failed"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Extract test counts
|
||||||
|
test_line=$(echo "$output" | grep -E "[0-9]+ tests?, [0-9]+ failures?" | head -1 || echo "0 tests, 0 failures")
|
||||||
|
total_tests=$(echo "$test_line" | grep -o '[0-9]\+ tests\?' | grep -o '[0-9]\+' | head -1 || echo "0")
|
||||||
|
failures=$(echo "$test_line" | grep -o '[0-9]\+ failures\?' | grep -o '[0-9]\+' | head -1 || echo "0")
|
||||||
|
|
||||||
|
echo "total=$total_tests" >> $GITHUB_OUTPUT
|
||||||
|
echo "failures=$failures" >> $GITHUB_OUTPUT
|
||||||
|
echo "passed=$((total_tests - failures))" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Calculate success rate
|
||||||
|
if [ "$total_tests" -gt 0 ]; then
|
||||||
|
success_rate=$(echo "scale=1; ($total_tests - $failures) * 100 / $total_tests" | bc)
|
||||||
|
else
|
||||||
|
success_rate="0"
|
||||||
|
fi
|
||||||
|
echo "success_rate=$success_rate" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
exit_code=$?
|
||||||
|
echo "exit_code=$exit_code" >> $GITHUB_OUTPUT
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Generate coverage report
|
||||||
|
id: coverage
|
||||||
|
run: |
|
||||||
|
# Generate coverage report with GitHub format
|
||||||
|
output=$(mix coveralls.github 2>&1 || true)
|
||||||
|
echo "$output" > coverage_output.txt
|
||||||
|
|
||||||
|
# Extract coverage percentage
|
||||||
|
coverage=$(echo "$output" | grep -o '[0-9]\+\.[0-9]\+%' | head -1 | sed 's/%//' || echo "0")
|
||||||
|
if [ -z "$coverage" ]; then
|
||||||
|
coverage="0"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "percentage=$coverage" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Determine status
|
||||||
|
if (( $(echo "$coverage >= 80" | bc -l) )); then
|
||||||
|
echo "status=✅ Excellent" >> $GITHUB_OUTPUT
|
||||||
|
elif (( $(echo "$coverage >= 60" | bc -l) )); then
|
||||||
|
echo "status=⚠️ Good" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Needs Improvement" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Run Credo analysis
|
||||||
|
id: credo
|
||||||
|
run: |
|
||||||
|
# Run Credo and capture output
|
||||||
|
output=$(mix credo --strict --format=json 2>&1 || true)
|
||||||
|
echo "$output" > credo_output.txt
|
||||||
|
|
||||||
|
# Try to parse JSON output
|
||||||
|
if echo "$output" | jq . > /dev/null 2>&1; then
|
||||||
|
issues=$(echo "$output" | jq '.issues | length' 2>/dev/null || echo "0")
|
||||||
|
high_issues=$(echo "$output" | jq '.issues | map(select(.priority == "high")) | length' 2>/dev/null || echo "0")
|
||||||
|
normal_issues=$(echo "$output" | jq '.issues | map(select(.priority == "normal")) | length' 2>/dev/null || echo "0")
|
||||||
|
low_issues=$(echo "$output" | jq '.issues | map(select(.priority == "low")) | length' 2>/dev/null || echo "0")
|
||||||
|
else
|
||||||
|
# Fallback: try to count issues from regular output
|
||||||
|
regular_output=$(mix credo --strict 2>&1 || true)
|
||||||
|
issues=$(echo "$regular_output" | grep -c "┃" || echo "0")
|
||||||
|
high_issues="0"
|
||||||
|
normal_issues="0"
|
||||||
|
low_issues="0"
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "total_issues=$issues" >> $GITHUB_OUTPUT
|
||||||
|
echo "high_issues=$high_issues" >> $GITHUB_OUTPUT
|
||||||
|
echo "normal_issues=$normal_issues" >> $GITHUB_OUTPUT
|
||||||
|
echo "low_issues=$low_issues" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Determine status
|
||||||
|
if [ "$issues" -eq 0 ]; then
|
||||||
|
echo "status=✅ Clean" >> $GITHUB_OUTPUT
|
||||||
|
elif [ "$issues" -lt 10 ]; then
|
||||||
|
echo "status=⚠️ Minor Issues" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Needs Attention" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Run Dialyzer analysis
|
||||||
|
id: dialyzer
|
||||||
|
run: |
|
||||||
|
# Ensure PLT is built
|
||||||
|
mix dialyzer --plt
|
||||||
|
|
||||||
|
# Run Dialyzer and capture output
|
||||||
|
output=$(mix dialyzer --format=github 2>&1 || true)
|
||||||
|
echo "$output" > dialyzer_output.txt
|
||||||
|
|
||||||
|
# Count warnings and errors
|
||||||
|
warnings=$(echo "$output" | grep -c "warning:" || echo "0")
|
||||||
|
errors=$(echo "$output" | grep -c "error:" || echo "0")
|
||||||
|
|
||||||
|
echo "warnings=$warnings" >> $GITHUB_OUTPUT
|
||||||
|
echo "errors=$errors" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Determine status
|
||||||
|
if [ "$errors" -eq 0 ] && [ "$warnings" -eq 0 ]; then
|
||||||
|
echo "status=✅ Clean" >> $GITHUB_OUTPUT
|
||||||
|
elif [ "$errors" -eq 0 ]; then
|
||||||
|
echo "status=⚠️ Warnings Only" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "status=❌ Has Errors" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Create test results summary
|
||||||
|
id: summary
|
||||||
|
run: |
|
||||||
|
# Calculate overall score
|
||||||
|
format_score=${{ steps.format.outputs.count == '0' && '100' || '0' }}
|
||||||
|
compile_score=${{ steps.compile.outputs.warnings == '0' && '100' || '80' }}
|
||||||
|
test_score=${{ steps.tests.outputs.success_rate }}
|
||||||
|
coverage_score=${{ steps.coverage.outputs.percentage }}
|
||||||
|
credo_score=$(echo "scale=0; (100 - ${{ steps.credo.outputs.total_issues }} * 2)" | bc | sed 's/^-.*$/0/')
|
||||||
|
dialyzer_score=$(echo "scale=0; (100 - ${{ steps.dialyzer.outputs.warnings }} * 2 - ${{ steps.dialyzer.outputs.errors }} * 10)" | bc | sed 's/^-.*$/0/')
|
||||||
|
|
||||||
|
overall_score=$(echo "scale=1; ($format_score + $compile_score + $test_score + $coverage_score + $credo_score + $dialyzer_score) / 6" | bc)
|
||||||
|
|
||||||
|
echo "overall_score=$overall_score" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
# Determine overall status
|
||||||
|
if (( $(echo "$overall_score >= 90" | bc -l) )); then
|
||||||
|
echo "overall_status=🌟 Excellent" >> $GITHUB_OUTPUT
|
||||||
|
elif (( $(echo "$overall_score >= 80" | bc -l) )); then
|
||||||
|
echo "overall_status=✅ Good" >> $GITHUB_OUTPUT
|
||||||
|
elif (( $(echo "$overall_score >= 70" | bc -l) )); then
|
||||||
|
echo "overall_status=⚠️ Needs Improvement" >> $GITHUB_OUTPUT
|
||||||
|
else
|
||||||
|
echo "overall_status=❌ Poor" >> $GITHUB_OUTPUT
|
||||||
|
fi
|
||||||
|
continue-on-error: true
|
||||||
|
|
||||||
|
- name: Find existing PR comment
|
||||||
|
if: github.event_name == 'pull_request'
|
||||||
|
id: find_comment
|
||||||
|
uses: peter-evans/find-comment@v3
|
||||||
|
with:
|
||||||
|
issue-number: ${{ github.event.pull_request.number }}
|
||||||
|
comment-author: 'github-actions[bot]'
|
||||||
|
body-includes: '## 🧪 Test Results Summary'
|
||||||
|
|
||||||
|
- name: Create or update PR comment
|
||||||
|
if: github.event_name == 'pull_request'
|
||||||
|
uses: peter-evans/create-or-update-comment@v4
|
||||||
|
with:
|
||||||
|
comment-id: ${{ steps.find_comment.outputs.comment-id }}
|
||||||
|
issue-number: ${{ github.event.pull_request.number }}
|
||||||
|
edit-mode: replace
|
||||||
|
body: |
|
||||||
|
## 🧪 Test Results Summary
|
||||||
|
|
||||||
|
**Overall Quality Score: ${{ steps.summary.outputs.overall_score }}%** ${{ steps.summary.outputs.overall_status }}
|
||||||
|
|
||||||
|
### 📊 Metrics Dashboard
|
||||||
|
|
||||||
|
| Category | Status | Count | Details |
|
||||||
|
|----------|---------|-------|---------|
|
||||||
|
| 📝 **Code Formatting** | ${{ steps.format.outputs.status }} | ${{ steps.format.outputs.count }} issues | `mix format --check-formatted` |
|
||||||
|
| 🔨 **Compilation** | ${{ steps.compile.outputs.status }} | ${{ steps.compile.outputs.warnings }} warnings | `mix compile` |
|
||||||
|
| 🧪 **Tests** | ${{ steps.tests.outputs.status }} | ${{ steps.tests.outputs.failures }}/${{ steps.tests.outputs.total }} failed | Success rate: ${{ steps.tests.outputs.success_rate }}% |
|
||||||
|
| 📊 **Coverage** | ${{ steps.coverage.outputs.status }} | ${{ steps.coverage.outputs.percentage }}% | `mix coveralls` |
|
||||||
|
| 🎯 **Credo** | ${{ steps.credo.outputs.status }} | ${{ steps.credo.outputs.total_issues }} issues | High: ${{ steps.credo.outputs.high_issues }}, Normal: ${{ steps.credo.outputs.normal_issues }}, Low: ${{ steps.credo.outputs.low_issues }} |
|
||||||
|
| 🔍 **Dialyzer** | ${{ steps.dialyzer.outputs.status }} | ${{ steps.dialyzer.outputs.errors }} errors, ${{ steps.dialyzer.outputs.warnings }} warnings | `mix dialyzer` |
|
||||||
|
|
||||||
|
### 🎯 Quality Gates
|
||||||
|
|
||||||
|
Based on the project's quality thresholds:
|
||||||
|
- **Compilation Warnings**: ${{ steps.compile.outputs.warnings }}/148 (limit: 148)
|
||||||
|
- **Credo Issues**: ${{ steps.credo.outputs.total_issues }}/87 (limit: 87)
|
||||||
|
- **Dialyzer Warnings**: ${{ steps.dialyzer.outputs.warnings }}/161 (limit: 161)
|
||||||
|
- **Test Coverage**: ${{ steps.coverage.outputs.percentage }}%/50% (minimum: 50%)
|
||||||
|
- **Test Failures**: ${{ steps.tests.outputs.failures }}/0 (limit: 0)
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>📈 Progress Toward Goals</summary>
|
||||||
|
|
||||||
|
Target goals for the project:
|
||||||
|
- ✨ **Zero compilation warnings** (currently: ${{ steps.compile.outputs.warnings }})
|
||||||
|
- ✨ **≤10 Credo issues** (currently: ${{ steps.credo.outputs.total_issues }})
|
||||||
|
- ✨ **Zero Dialyzer warnings** (currently: ${{ steps.dialyzer.outputs.warnings }})
|
||||||
|
- ✨ **≥85% test coverage** (currently: ${{ steps.coverage.outputs.percentage }}%)
|
||||||
|
- ✅ **Zero test failures** (currently: ${{ steps.tests.outputs.failures }})
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
<details>
|
||||||
|
<summary>🔧 Quick Actions</summary>
|
||||||
|
|
||||||
|
To improve code quality:
|
||||||
|
```bash
|
||||||
|
# Fix formatting issues
|
||||||
|
mix format
|
||||||
|
|
||||||
|
# View detailed Credo analysis
|
||||||
|
mix credo --strict
|
||||||
|
|
||||||
|
# Check Dialyzer warnings
|
||||||
|
mix dialyzer
|
||||||
|
|
||||||
|
# Generate detailed coverage report
|
||||||
|
mix coveralls.html
|
||||||
|
```
|
||||||
|
|
||||||
|
</details>
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
🤖 *Auto-generated by GitHub Actions* • Updated: ${{ github.event.head_commit.timestamp }}
|
||||||
|
|
||||||
|
> **Note**: This comment will be updated automatically when new commits are pushed to this PR.
|
||||||
6
.gitignore
vendored
6
.gitignore
vendored
@@ -4,7 +4,8 @@
|
|||||||
*.iml
|
*.iml
|
||||||
|
|
||||||
*.key
|
*.key
|
||||||
|
.repomixignore
|
||||||
|
repomix*
|
||||||
/.idea/
|
/.idea/
|
||||||
/node_modules/
|
/node_modules/
|
||||||
/assets/node_modules/
|
/assets/node_modules/
|
||||||
@@ -17,6 +18,9 @@
|
|||||||
/priv/static/*.js
|
/priv/static/*.js
|
||||||
/priv/static/*.css
|
/priv/static/*.css
|
||||||
|
|
||||||
|
# Dialyzer PLT files
|
||||||
|
/priv/plts/
|
||||||
|
|
||||||
.DS_Store
|
.DS_Store
|
||||||
**/.DS_Store
|
**/.DS_Store
|
||||||
|
|
||||||
|
|||||||
269
CHANGELOG.md
269
CHANGELOG.md
@@ -2,6 +2,275 @@
|
|||||||
|
|
||||||
<!-- changelog -->
|
<!-- changelog -->
|
||||||
|
|
||||||
|
## [v1.75.3](https://github.com/wanderer-industries/wanderer/compare/v1.75.2...v1.75.3) (2025-08-10)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* core: Fixed character tracking issues
|
||||||
|
|
||||||
|
## [v1.75.2](https://github.com/wanderer-industries/wanderer/compare/v1.75.1...v1.75.2) (2025-08-10)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix indents for ally logos in list "On the map"
|
||||||
|
|
||||||
|
* Map: Fix cancelling ping from system context menu
|
||||||
|
|
||||||
|
* Map: Hide admin settings tab
|
||||||
|
|
||||||
|
* Map: Remote map setting refactoring
|
||||||
|
|
||||||
|
## [v1.75.1](https://github.com/wanderer-industries/wanderer/compare/v1.75.0...v1.75.1) (2025-07-30)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* unable to cancel ping from right click context menu
|
||||||
|
|
||||||
|
## [v1.75.0](https://github.com/wanderer-industries/wanderer/compare/v1.74.13...v1.75.0) (2025-07-29)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* autoset connection size for c4->null and c13
|
||||||
|
|
||||||
|
* apiv1 and tests
|
||||||
|
|
||||||
|
* support webhook and sse
|
||||||
|
|
||||||
|
* disable webhook/websocket by default
|
||||||
|
|
||||||
|
* add websocket and webhooks for events
|
||||||
|
|
||||||
|
* Add Jest testing for getState util
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* remove bug with lazy delete
|
||||||
|
|
||||||
|
* update broken length and remove verbose logging
|
||||||
|
|
||||||
|
* removed old documents
|
||||||
|
|
||||||
|
* removed unneeded api, and fixed data comparision bug
|
||||||
|
|
||||||
|
* ci comments
|
||||||
|
|
||||||
|
* test updates
|
||||||
|
|
||||||
|
* properly send sse events
|
||||||
|
|
||||||
|
* add test coverage for api
|
||||||
|
|
||||||
|
* add more logging around character online and tracking
|
||||||
|
|
||||||
|
* clean up SSE warnings
|
||||||
|
|
||||||
|
* update env variable usage for sse
|
||||||
|
|
||||||
|
* sse cleanup
|
||||||
|
|
||||||
|
* remove misleading error
|
||||||
|
|
||||||
|
* update killactivity color on nodes
|
||||||
|
|
||||||
|
## [v1.74.13](https://github.com/wanderer-industries/wanderer/compare/v1.74.12...v1.74.13) (2025-07-29)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Core: Fixed issue with callback url
|
||||||
|
|
||||||
|
## [v1.74.12](https://github.com/wanderer-industries/wanderer/compare/v1.74.11...v1.74.12) (2025-07-22)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## [v1.74.11](https://github.com/wanderer-industries/wanderer/compare/v1.74.10...v1.74.11) (2025-07-18)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fixed remove pings for removed systems
|
||||||
|
|
||||||
|
## [v1.74.10](https://github.com/wanderer-industries/wanderer/compare/v1.74.9...v1.74.10) (2025-07-15)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## [v1.74.9](https://github.com/wanderer-industries/wanderer/compare/v1.74.8...v1.74.9) (2025-07-13)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Trying to fix problem with fast forwarding after page are inactive some time.
|
||||||
|
|
||||||
|
## [v1.74.8](https://github.com/wanderer-industries/wanderer/compare/v1.74.7...v1.74.8) (2025-07-11)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: removed comments
|
||||||
|
|
||||||
|
* Map: Fixed conflict
|
||||||
|
|
||||||
|
* Map: Unified settings. Second part: Import/Export
|
||||||
|
|
||||||
|
* Map: Unified settings. First part: add one place for storing settings
|
||||||
|
|
||||||
|
## [v1.74.7](https://github.com/wanderer-industries/wanderer/compare/v1.74.6...v1.74.7) (2025-07-09)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## [v1.74.6](https://github.com/wanderer-industries/wanderer/compare/v1.74.5...v1.74.6) (2025-07-09)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
## [v1.74.5](https://github.com/wanderer-industries/wanderer/compare/v1.74.4...v1.74.5) (2025-07-09)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Add background for Pochven's systems. Changed from Region name to constellation name for pochven systems. Changed connection style for gates (display like common connection). Changed behaviour of connections.
|
||||||
|
|
||||||
|
## [v1.74.4](https://github.com/wanderer-industries/wanderer/compare/v1.74.3...v1.74.4) (2025-07-07)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Core: Fixed issue with update system positions
|
||||||
|
|
||||||
|
## [v1.74.3](https://github.com/wanderer-industries/wanderer/compare/v1.74.2...v1.74.3) (2025-07-06)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Core: Fixed issues with map subscription component
|
||||||
|
|
||||||
|
## [v1.74.2](https://github.com/wanderer-industries/wanderer/compare/v1.74.1...v1.74.2) (2025-06-30)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Core: Fixed map loading for not existing maps
|
||||||
|
|
||||||
|
## [v1.74.1](https://github.com/wanderer-industries/wanderer/compare/v1.74.0...v1.74.1) (2025-06-28)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Core: Mark connections between Pochven systems as known.
|
||||||
|
|
||||||
|
## [v1.74.0](https://github.com/wanderer-industries/wanderer/compare/v1.73.0...v1.74.0) (2025-06-25)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Core: Reverted showing linked signature ID as part of temporary names
|
||||||
|
|
||||||
|
## [v1.73.0](https://github.com/wanderer-industries/wanderer/compare/v1.72.1...v1.73.0) (2025-06-25)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Core: Allowed system temp names up to 12 characters. Deprecated showing linked signature ID as part of temporary name.
|
||||||
|
|
||||||
|
## [v1.72.1](https://github.com/wanderer-industries/wanderer/compare/v1.72.0...v1.72.1) (2025-06-23)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* issue with tracking signature activity
|
||||||
|
|
||||||
|
## [v1.72.0](https://github.com/wanderer-industries/wanderer/compare/v1.71.3...v1.72.0) (2025-06-21)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* Core: Added an ability to see & topup map balance and map subscription info (on public)
|
||||||
|
|
||||||
|
## [v1.71.3](https://github.com/wanderer-industries/wanderer/compare/v1.71.2...v1.71.3) (2025-06-21)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* Map: Fix incorrect placing of labels
|
||||||
|
|
||||||
|
## [v1.71.2](https://github.com/wanderer-industries/wanderer/compare/v1.71.1...v1.71.2) (2025-06-20)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* fix issue with kill service disconnect
|
||||||
|
|
||||||
|
## [v1.71.1](https://github.com/wanderer-industries/wanderer/compare/v1.71.0...v1.71.1) (2025-06-19)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* update system kills widget timing
|
||||||
|
|
||||||
|
## [v1.71.0](https://github.com/wanderer-industries/wanderer/compare/v1.70.7...v1.71.0) (2025-06-19)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### Features:
|
||||||
|
|
||||||
|
* use external services for kill data
|
||||||
|
|
||||||
|
### Bug Fixes:
|
||||||
|
|
||||||
|
* remove duplicate kills connections
|
||||||
|
|
||||||
|
* Fixed kills clinet init & map start/update logic
|
||||||
|
|
||||||
|
* avoid duplicate subs, and remove subs on inactive maps
|
||||||
|
|
||||||
## [v1.70.7](https://github.com/wanderer-industries/wanderer/compare/v1.70.6...v1.70.7) (2025-06-18)
|
## [v1.70.7](https://github.com/wanderer-industries/wanderer/compare/v1.70.6...v1.70.7) (2025-06-18)
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
14
assets/jest.config.js
Normal file
14
assets/jest.config.js
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
module.exports = {
|
||||||
|
preset: 'ts-jest',
|
||||||
|
testEnvironment: 'jsdom',
|
||||||
|
roots: ['<rootDir>'],
|
||||||
|
moduleDirectories: ['node_modules', 'js'],
|
||||||
|
moduleNameMapper: {
|
||||||
|
'^@/(.*)$': '<rootDir>/js/$1',
|
||||||
|
'\.scss$': 'identity-obj-proxy', // Mock SCSS files
|
||||||
|
},
|
||||||
|
transform: {
|
||||||
|
'^.+\.(ts|tsx)$': 'ts-jest',
|
||||||
|
'^.+\.(js|jsx)$': 'babel-jest', // Add babel-jest for JS/JSX files if needed
|
||||||
|
},
|
||||||
|
};
|
||||||
@@ -212,3 +212,75 @@
|
|||||||
.p-inputtext:enabled:hover {
|
.p-inputtext:enabled:hover {
|
||||||
border-color: #335c7e;
|
border-color: #335c7e;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// --------------- TOAST
|
||||||
|
.p-toast .p-toast-message {
|
||||||
|
background-color: #1a1a1a;
|
||||||
|
color: #e0e0e0;
|
||||||
|
border-left: 4px solid transparent;
|
||||||
|
border-radius: 4px;
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.7);
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast .p-toast-message .p-toast-summary {
|
||||||
|
color: #ffffff;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast .p-toast-message .p-toast-detail {
|
||||||
|
color: #c0c0c0;
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast .p-toast-icon-close {
|
||||||
|
color: #ffaa00;
|
||||||
|
transition: background 0.2s;
|
||||||
|
}
|
||||||
|
.p-toast .p-toast-icon-close:hover {
|
||||||
|
background: #333;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-success {
|
||||||
|
border-left-color: #f1c40f;
|
||||||
|
}
|
||||||
|
.p-toast-message-error {
|
||||||
|
border-left-color: #e74c3c;
|
||||||
|
}
|
||||||
|
.p-toast-message-info {
|
||||||
|
border-left-color: #3498db;
|
||||||
|
}
|
||||||
|
.p-toast-message-warn {
|
||||||
|
border-left-color: #e67e22;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-success .p-toast-message-icon {
|
||||||
|
color: #f1c40f;
|
||||||
|
}
|
||||||
|
.p-toast-message-error .p-toast-message-icon {
|
||||||
|
color: #e74c3c;
|
||||||
|
}
|
||||||
|
.p-toast-message-info .p-toast-message-icon {
|
||||||
|
color: #3498db;
|
||||||
|
}
|
||||||
|
.p-toast-message-warn .p-toast-message-icon {
|
||||||
|
color: #e67e22;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-success .p-toast-message-content {
|
||||||
|
border-left-color: #f1c40f;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-error .p-toast-message-content {
|
||||||
|
border-left-color: #e74c3c;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-info .p-toast-message-content {
|
||||||
|
border-left-color: #3498db;
|
||||||
|
}
|
||||||
|
|
||||||
|
.p-toast-message-warn .p-toast-message-content {
|
||||||
|
border-left-color: #e67e22;
|
||||||
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -64,9 +64,9 @@ body .p-dialog {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.p-dialog-footer {
|
.p-dialog-footer {
|
||||||
padding: 1rem;
|
padding: .75rem 1rem;
|
||||||
border-top: 1px solid #ddd;
|
border-top: none !important;
|
||||||
background: #f4f4f4;
|
//background: #f4f4f4;
|
||||||
}
|
}
|
||||||
|
|
||||||
.p-dialog-header-close {
|
.p-dialog-header-close {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
.vertical-tabs-container {
|
.vertical-tabs-container {
|
||||||
display: flex;
|
display: flex;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
min-height: 300px;
|
min-height: 400px;
|
||||||
|
|
||||||
.p-tabview {
|
.p-tabview {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@@ -68,6 +68,28 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.color-warn {
|
||||||
|
@apply bg-yellow-600/5 border-r-yellow-600/20;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
@apply bg-yellow-600/10 border-r-yellow-600/40;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
&.p-tabview-selected {
|
||||||
|
@apply bg-yellow-600/10 border-r-yellow-600;
|
||||||
|
|
||||||
|
.p-tabview-nav-link {
|
||||||
|
@apply text-yellow-600;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
@apply bg-yellow-600/10 border-r-yellow-600;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ export interface ContextMenuSystemProps {
|
|||||||
onSystemStatus(val: number): void;
|
onSystemStatus(val: number): void;
|
||||||
onSystemLabels(val: string): void;
|
onSystemLabels(val: string): void;
|
||||||
onCustomLabelDialog(): void;
|
onCustomLabelDialog(): void;
|
||||||
onTogglePing(type: PingType, solar_system_id: string, hasPing: boolean): void;
|
onTogglePing(type: PingType, solar_system_id: string, ping_id: string | undefined, hasPing: boolean): void;
|
||||||
onWaypointSet: WaypointSetContextHandler;
|
onWaypointSet: WaypointSetContextHandler;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -109,7 +109,7 @@ export const useContextMenuSystemItems = ({
|
|||||||
|
|
||||||
{ separator: true },
|
{ separator: true },
|
||||||
{
|
{
|
||||||
command: () => onTogglePing(PingType.Rally, systemId, hasPing),
|
command: () => onTogglePing(PingType.Rally, systemId, ping?.id, hasPing),
|
||||||
disabled: !isShowPingBtn,
|
disabled: !isShowPingBtn,
|
||||||
template: () => {
|
template: () => {
|
||||||
const iconClasses = clsx({
|
const iconClasses = clsx({
|
||||||
|
|||||||
@@ -1,17 +1,24 @@
|
|||||||
import { Node } from 'reactflow';
|
import { Node } from 'reactflow';
|
||||||
import { useCallback, useRef, useState } from 'react';
|
import { useCallback, useMemo, useRef, useState } from 'react';
|
||||||
import { ContextMenu } from 'primereact/contextmenu';
|
import { ContextMenu } from 'primereact/contextmenu';
|
||||||
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
import { SolarSystemRawType } from '@/hooks/Mapper/types';
|
||||||
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
|
||||||
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
import { NodeSelectionMouseHandler } from '@/hooks/Mapper/components/contexts/types.ts';
|
||||||
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
import { useDeleteSystems } from '@/hooks/Mapper/components/contexts/hooks';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
export const useContextMenuSystemMultipleHandlers = () => {
|
export const useContextMenuSystemMultipleHandlers = () => {
|
||||||
|
const {
|
||||||
|
data: { pings },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
const contextMenuRef = useRef<ContextMenu | null>(null);
|
const contextMenuRef = useRef<ContextMenu | null>(null);
|
||||||
const [systems, setSystems] = useState<Node<SolarSystemRawType>[]>();
|
const [systems, setSystems] = useState<Node<SolarSystemRawType>[]>();
|
||||||
|
|
||||||
const { deleteSystems } = useDeleteSystems();
|
const { deleteSystems } = useDeleteSystems();
|
||||||
|
|
||||||
|
const ping = useMemo(() => (pings.length === 1 ? pings[0] : undefined), [pings]);
|
||||||
|
|
||||||
const handleSystemMultipleContext: NodeSelectionMouseHandler = (ev, systems_) => {
|
const handleSystemMultipleContext: NodeSelectionMouseHandler = (ev, systems_) => {
|
||||||
setSystems(systems_);
|
setSystems(systems_);
|
||||||
ev.preventDefault();
|
ev.preventDefault();
|
||||||
@@ -24,13 +31,17 @@ export const useContextMenuSystemMultipleHandlers = () => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const sysToDel = systems.filter(x => !x.data.locked).map(x => x.id);
|
const sysToDel = systems
|
||||||
|
.filter(x => !x.data.locked)
|
||||||
|
.filter(x => x.id !== ping?.solar_system_id)
|
||||||
|
.map(x => x.id);
|
||||||
|
|
||||||
if (sysToDel.length === 0) {
|
if (sysToDel.length === 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
deleteSystems(sysToDel);
|
deleteSystems(sysToDel);
|
||||||
}, [deleteSystems, systems]);
|
}, [deleteSystems, systems, ping]);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
handleSystemMultipleContext,
|
handleSystemMultipleContext,
|
||||||
|
|||||||
1
assets/js/hooks/Mapper/components/helpers/index.ts
Normal file
1
assets/js/hooks/Mapper/components/helpers/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './parseMapUserSettings.ts';
|
||||||
@@ -0,0 +1,67 @@
|
|||||||
|
import { MapUserSettings, SettingsWithVersion } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
|
||||||
|
export const REQUIRED_KEYS = [
|
||||||
|
'widgets',
|
||||||
|
'interface',
|
||||||
|
'onTheMap',
|
||||||
|
'routes',
|
||||||
|
'localWidget',
|
||||||
|
'signaturesWidget',
|
||||||
|
'killsWidget',
|
||||||
|
] as const;
|
||||||
|
|
||||||
|
type RequiredKeys = (typeof REQUIRED_KEYS)[number];
|
||||||
|
|
||||||
|
/** Custom error for any parsing / validation issue */
|
||||||
|
export class MapUserSettingsParseError extends Error {
|
||||||
|
constructor(msg: string) {
|
||||||
|
super(`MapUserSettings parse error: ${msg}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const isNumber = (v: unknown): v is number => typeof v === 'number' && !Number.isNaN(v);
|
||||||
|
|
||||||
|
/** Minimal check that an object matches SettingsWithVersion<*> */
|
||||||
|
const isSettingsWithVersion = (v: unknown): v is SettingsWithVersion<unknown> =>
|
||||||
|
typeof v === 'object' && v !== null && isNumber((v as any).version) && 'settings' in (v as any);
|
||||||
|
|
||||||
|
/** Ensure every required key is present */
|
||||||
|
const hasAllRequiredKeys = (v: unknown): v is Record<RequiredKeys, unknown> =>
|
||||||
|
typeof v === 'object' && v !== null && REQUIRED_KEYS.every(k => k in v);
|
||||||
|
|
||||||
|
/* ------------------------------ Main parser ------------------------------- */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses and validates a JSON string as `MapUserSettings`.
|
||||||
|
*
|
||||||
|
* @throws `MapUserSettingsParseError` – если строка не JSON или нарушена структура
|
||||||
|
*/
|
||||||
|
export const parseMapUserSettings = (json: unknown): MapUserSettings => {
|
||||||
|
if (typeof json !== 'string') throw new MapUserSettingsParseError('Input must be a JSON string');
|
||||||
|
|
||||||
|
let data: unknown;
|
||||||
|
try {
|
||||||
|
data = JSON.parse(json);
|
||||||
|
} catch (e) {
|
||||||
|
throw new MapUserSettingsParseError(`Invalid JSON: ${(e as Error).message}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!hasAllRequiredKeys(data)) {
|
||||||
|
const missing = REQUIRED_KEYS.filter(k => !(k in (data as any)));
|
||||||
|
throw new MapUserSettingsParseError(`Missing top-level field(s): ${missing.join(', ')}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const key of REQUIRED_KEYS) {
|
||||||
|
if (!isSettingsWithVersion((data as any)[key])) {
|
||||||
|
throw new MapUserSettingsParseError(`"${key}" must match SettingsWithVersion<T>`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Everything passes, so cast is safe
|
||||||
|
return data as MapUserSettings;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* ------------------------------ Usage example ----------------------------- */
|
||||||
|
|
||||||
|
// const raw = fetchFromServer(); // string
|
||||||
|
// const settings = parseMapUserSettings(raw);
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
export * from './useSystemInfo';
|
export * from './useSystemInfo';
|
||||||
export * from './useGetOwnOnlineCharacters';
|
export * from './useGetOwnOnlineCharacters';
|
||||||
export * from './useElementWidth';
|
export * from './useElementWidth';
|
||||||
|
export * from './useDetectSettingsChanged';
|
||||||
|
|||||||
@@ -0,0 +1,23 @@
|
|||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
export const useDetectSettingsChanged = () => {
|
||||||
|
const {
|
||||||
|
storedSettings: {
|
||||||
|
interfaceSettings,
|
||||||
|
settingsRoutes,
|
||||||
|
settingsLocal,
|
||||||
|
settingsSignatures,
|
||||||
|
settingsOnTheMap,
|
||||||
|
settingsKills,
|
||||||
|
},
|
||||||
|
} = useMapRootState();
|
||||||
|
const [counter, setCounter] = useState(0);
|
||||||
|
|
||||||
|
useEffect(
|
||||||
|
() => setCounter(x => x + 1),
|
||||||
|
[interfaceSettings, settingsRoutes, settingsLocal, settingsSignatures, settingsOnTheMap, settingsKills],
|
||||||
|
);
|
||||||
|
|
||||||
|
return counter;
|
||||||
|
};
|
||||||
@@ -98,6 +98,7 @@ interface MapCompProps {
|
|||||||
theme?: string;
|
theme?: string;
|
||||||
pings: PingData[];
|
pings: PingData[];
|
||||||
minimapPlacement?: PanelPosition;
|
minimapPlacement?: PanelPosition;
|
||||||
|
localShowShipName?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const MapComp = ({
|
const MapComp = ({
|
||||||
@@ -117,6 +118,7 @@ const MapComp = ({
|
|||||||
onAddSystem,
|
onAddSystem,
|
||||||
pings,
|
pings,
|
||||||
minimapPlacement = 'bottom-right',
|
minimapPlacement = 'bottom-right',
|
||||||
|
localShowShipName = false,
|
||||||
}: MapCompProps) => {
|
}: MapCompProps) => {
|
||||||
const { getNodes } = useReactFlow();
|
const { getNodes } = useReactFlow();
|
||||||
const [nodes, , onNodesChange] = useNodesState<Node<SolarSystemRawType>>(initialNodes);
|
const [nodes, , onNodesChange] = useNodesState<Node<SolarSystemRawType>>(initialNodes);
|
||||||
@@ -212,8 +214,9 @@ const MapComp = ({
|
|||||||
showKSpaceBG: showKSpaceBG,
|
showKSpaceBG: showKSpaceBG,
|
||||||
isThickConnections: isThickConnections,
|
isThickConnections: isThickConnections,
|
||||||
pings,
|
pings,
|
||||||
|
localShowShipName,
|
||||||
}));
|
}));
|
||||||
}, [showKSpaceBG, isThickConnections, pings, update]);
|
}, [showKSpaceBG, isThickConnections, pings, update, localShowShipName]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ export type MapData = MapUnionTypes & {
|
|||||||
showKSpaceBG: boolean;
|
showKSpaceBG: boolean;
|
||||||
isThickConnections: boolean;
|
isThickConnections: boolean;
|
||||||
linkedSigEveId: string;
|
linkedSigEveId: string;
|
||||||
|
localShowShipName: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
interface MapProviderProps {
|
interface MapProviderProps {
|
||||||
@@ -42,6 +43,7 @@ const INITIAL_DATA: MapData = {
|
|||||||
followingCharacterEveId: null,
|
followingCharacterEveId: null,
|
||||||
userHubs: [],
|
userHubs: [],
|
||||||
pings: [],
|
pings: [],
|
||||||
|
localShowShipName: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export interface MapContextProps {
|
export interface MapContextProps {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { useMemo } from 'react';
|
import { useMemo } from 'react';
|
||||||
import { useKillsCounter } from '../../hooks/useKillsCounter';
|
import { useKillsCounter } from '../../hooks/useKillsCounter.ts';
|
||||||
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
||||||
import { WithChildren, WithClassName } from '@/hooks/Mapper/types/common';
|
import { WithChildren, WithClassName } from '@/hooks/Mapper/types/common.ts';
|
||||||
import {
|
import {
|
||||||
KILLS_ROW_HEIGHT,
|
KILLS_ROW_HEIGHT,
|
||||||
SystemKillsList,
|
SystemKillsList,
|
||||||
@@ -49,7 +49,7 @@ export const KillsCounter = ({
|
|||||||
content={
|
content={
|
||||||
<div className="overflow-hidden flex w-[450px] flex-col" style={{ height: `${tooltipHeight}px` }}>
|
<div className="overflow-hidden flex w-[450px] flex-col" style={{ height: `${tooltipHeight}px` }}>
|
||||||
<div className="flex-1 h-full">
|
<div className="flex-1 h-full">
|
||||||
<SystemKillsList kills={limitedKills} onlyOneSystem />
|
<SystemKillsList kills={limitedKills} onlyOneSystem timeRange={1} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
}
|
}
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './KillsCounter.tsx';
|
||||||
@@ -3,11 +3,11 @@ import clsx from 'clsx';
|
|||||||
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
import { WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper';
|
||||||
import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit/WdTooltip';
|
import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit/WdTooltip';
|
||||||
import { CharItemProps, LocalCharactersList } from '../../../mapInterface/widgets/LocalCharacters/components';
|
import { CharItemProps, LocalCharactersList } from '../../../mapInterface/widgets/LocalCharacters/components';
|
||||||
import { useLocalCharactersItemTemplate } from '../../../mapInterface/widgets/LocalCharacters/hooks/useLocalCharacters';
|
|
||||||
import { useLocalCharacterWidgetSettings } from '../../../mapInterface/widgets/LocalCharacters/hooks/useLocalWidgetSettings';
|
|
||||||
import classes from './SolarSystemLocalCounter.module.scss';
|
|
||||||
import { useTheme } from '@/hooks/Mapper/hooks/useTheme.ts';
|
import { useTheme } from '@/hooks/Mapper/hooks/useTheme.ts';
|
||||||
import { AvailableThemes } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
import { AvailableThemes } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import classes from './LocalCounter.module.scss';
|
||||||
|
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider.tsx';
|
||||||
|
import { useLocalCharactersItemTemplate } from '@/hooks/Mapper/components/mapInterface/widgets/LocalCharacters/hooks/useLocalCharacters.tsx';
|
||||||
|
|
||||||
interface LocalCounterProps {
|
interface LocalCounterProps {
|
||||||
localCounterCharacters: Array<CharItemProps>;
|
localCounterCharacters: Array<CharItemProps>;
|
||||||
@@ -16,8 +16,10 @@ interface LocalCounterProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const LocalCounter = ({ localCounterCharacters, hasUserCharacters, showIcon = true }: LocalCounterProps) => {
|
export const LocalCounter = ({ localCounterCharacters, hasUserCharacters, showIcon = true }: LocalCounterProps) => {
|
||||||
const [settings] = useLocalCharacterWidgetSettings();
|
const {
|
||||||
const itemTemplate = useLocalCharactersItemTemplate(settings.showShipName);
|
data: { localShowShipName },
|
||||||
|
} = useMapState();
|
||||||
|
const itemTemplate = useLocalCharactersItemTemplate(localShowShipName);
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
const pilotTooltipContent = useMemo(() => {
|
const pilotTooltipContent = useMemo(() => {
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
export * from './LocalCounter';
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useCallback, useMemo, useState } from 'react';
|
||||||
|
|
||||||
import classes from './SolarSystemEdge.module.scss';
|
import classes from './SolarSystemEdge.module.scss';
|
||||||
import { EdgeLabelRenderer, EdgeProps, getBezierPath, getSmoothStepPath, Position, useStore } from 'reactflow';
|
import { EdgeLabelRenderer, EdgeProps, getBezierPath, Position, useStore } from 'reactflow';
|
||||||
import { getEdgeParams } from '@/hooks/Mapper/components/map/utils.ts';
|
import { getEdgeParams } from '@/hooks/Mapper/components/map/utils.ts';
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import { ConnectionType, MassState, ShipSizeStatus, SolarSystemConnection, TimeStatus } from '@/hooks/Mapper/types';
|
import { ConnectionType, MassState, ShipSizeStatus, SolarSystemConnection, TimeStatus } from '@/hooks/Mapper/types';
|
||||||
@@ -51,11 +51,11 @@ export const SolarSystemEdge = ({ id, source, target, markerEnd, style, data }:
|
|||||||
const [hovered, setHovered] = useState(false);
|
const [hovered, setHovered] = useState(false);
|
||||||
|
|
||||||
const [path, labelX, labelY, sx, sy, tx, ty, sourcePos, targetPos] = useMemo(() => {
|
const [path, labelX, labelY, sx, sy, tx, ty, sourcePos, targetPos] = useMemo(() => {
|
||||||
const { sx, sy, tx, ty, sourcePos, targetPos } = getEdgeParams(sourceNode, targetNode);
|
const { sx, sy, tx, ty, sourcePos, targetPos } = getEdgeParams(sourceNode!, targetNode!);
|
||||||
|
|
||||||
const offset = isThickConnections ? MAP_OFFSETS_TICK[targetPos] : MAP_OFFSETS[targetPos];
|
const offset = isThickConnections ? MAP_OFFSETS_TICK[targetPos] : MAP_OFFSETS[targetPos];
|
||||||
|
|
||||||
const method = isWormhole ? getBezierPath : getSmoothStepPath;
|
const method = isWormhole ? getBezierPath : getBezierPath;
|
||||||
|
|
||||||
const [edgePath, labelX, labelY] = method({
|
const [edgePath, labelX, labelY] = method({
|
||||||
sourceX: sx - offset.x,
|
sourceX: sx - offset.x,
|
||||||
|
|||||||
@@ -40,6 +40,7 @@ $neon-color-3: rgba(27, 132, 236, 0.40);
|
|||||||
z-index: 3;
|
z-index: 3;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
|
||||||
|
&.Pochven,
|
||||||
&.Mataria,
|
&.Mataria,
|
||||||
&.Amarria,
|
&.Amarria,
|
||||||
&.Gallente,
|
&.Gallente,
|
||||||
@@ -95,6 +96,15 @@ $neon-color-3: rgba(27, 132, 236, 0.40);
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&.Pochven {
|
||||||
|
&::after {
|
||||||
|
opacity: 0.8;
|
||||||
|
background-image: url('/images/pochven.webp');
|
||||||
|
background-position-x: 0;
|
||||||
|
background-position-y: -13px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
&.selected {
|
&.selected {
|
||||||
border-color: $pastel-pink;
|
border-color: $pastel-pink;
|
||||||
box-shadow: 0 0 10px #9a1af1c2;
|
box-shadow: 0 0 10px #9a1af1c2;
|
||||||
|
|||||||
@@ -12,17 +12,19 @@ import {
|
|||||||
} from '@/hooks/Mapper/components/map/constants';
|
} from '@/hooks/Mapper/components/map/constants';
|
||||||
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
||||||
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
||||||
import { LocalCounter } from './SolarSystemLocalCounter';
|
|
||||||
import { KillsCounter } from './SolarSystemKillsCounter';
|
|
||||||
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
|
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
|
||||||
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
||||||
import { Tag } from 'primereact/tag';
|
import { Tag } from 'primereact/tag';
|
||||||
|
import { LocalCounter } from '@/hooks/Mapper/components/map/components/LocalCounter';
|
||||||
|
import { KillsCounter } from '@/hooks/Mapper/components/map/components/KillsCounter';
|
||||||
|
|
||||||
// let render = 0;
|
// let render = 0;
|
||||||
export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>) => {
|
export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>) => {
|
||||||
const nodeVars = useSolarSystemNode(props);
|
const nodeVars = useSolarSystemNode(props);
|
||||||
const { localCounterCharacters } = useLocalCounter(nodeVars);
|
const { localCounterCharacters } = useLocalCounter(nodeVars);
|
||||||
const localKillsCount = useNodeKillsCount(nodeVars.solarSystemId, nodeVars.killsCount);
|
const { killsCount: localKillsCount, killsActivityType: localKillsActivityType } = useNodeKillsCount(
|
||||||
|
nodeVars.solarSystemId,
|
||||||
|
);
|
||||||
|
|
||||||
// console.log('JOipP', `render ${nodeVars.id}`, render++);
|
// console.log('JOipP', `render ${nodeVars.id}`, render++);
|
||||||
|
|
||||||
@@ -38,17 +40,17 @@ export const SolarSystemNodeDefault = memo((props: NodeProps<MapSolarSystemType>
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{localKillsCount && localKillsCount > 0 && nodeVars.solarSystemId && (
|
{localKillsCount != null && localKillsCount > 0 && nodeVars.solarSystemId && localKillsActivityType && (
|
||||||
<KillsCounter
|
<KillsCounter
|
||||||
killsCount={localKillsCount}
|
killsCount={localKillsCount}
|
||||||
systemId={nodeVars.solarSystemId}
|
systemId={nodeVars.solarSystemId}
|
||||||
size={TooltipSize.lg}
|
size={TooltipSize.lg}
|
||||||
killsActivityType={nodeVars.killsActivityType}
|
killsActivityType={localKillsActivityType}
|
||||||
className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[nodeVars.killsActivityType!])}
|
className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[localKillsActivityType])}
|
||||||
>
|
>
|
||||||
<div className={clsx(classes.BookmarkWithIcon)}>
|
<div className={clsx(classes.BookmarkWithIcon)}>
|
||||||
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
||||||
<span className={clsx(classes.text)}>{nodeVars.killsCount}</span>
|
<span className={clsx(classes.text)}>{localKillsCount}</span>
|
||||||
</div>
|
</div>
|
||||||
</KillsCounter>
|
</KillsCounter>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -12,16 +12,18 @@ import {
|
|||||||
} from '@/hooks/Mapper/components/map/constants';
|
} from '@/hooks/Mapper/components/map/constants';
|
||||||
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
import { WormholeClassComp } from '@/hooks/Mapper/components/map/components/WormholeClassComp';
|
||||||
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
import { UnsplashedSignature } from '@/hooks/Mapper/components/map/components/UnsplashedSignature';
|
||||||
import { LocalCounter } from './SolarSystemLocalCounter';
|
|
||||||
import { KillsCounter } from './SolarSystemKillsCounter';
|
|
||||||
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
||||||
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
|
import { TooltipSize } from '@/hooks/Mapper/components/ui-kit/WdTooltipWrapper/utils.ts';
|
||||||
|
import { LocalCounter } from '@/hooks/Mapper/components/map/components/LocalCounter';
|
||||||
|
import { KillsCounter } from '@/hooks/Mapper/components/map/components/KillsCounter';
|
||||||
|
|
||||||
// let render = 0;
|
// let render = 0;
|
||||||
export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>) => {
|
export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>) => {
|
||||||
const nodeVars = useSolarSystemNode(props);
|
const nodeVars = useSolarSystemNode(props);
|
||||||
const { localCounterCharacters } = useLocalCounter(nodeVars);
|
const { localCounterCharacters } = useLocalCounter(nodeVars);
|
||||||
const localKillsCount = useNodeKillsCount(nodeVars.solarSystemId, nodeVars.killsCount);
|
const { killsCount: localKillsCount, killsActivityType: localKillsActivityType } = useNodeKillsCount(
|
||||||
|
nodeVars.solarSystemId,
|
||||||
|
);
|
||||||
|
|
||||||
// console.log('JOipP', `render ${nodeVars.id}`, render++);
|
// console.log('JOipP', `render ${nodeVars.id}`, render++);
|
||||||
|
|
||||||
@@ -37,17 +39,17 @@ export const SolarSystemNodeTheme = memo((props: NodeProps<MapSolarSystemType>)
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{localKillsCount && localKillsCount > 0 && nodeVars.solarSystemId && (
|
{localKillsCount && localKillsCount > 0 && nodeVars.solarSystemId && localKillsActivityType && (
|
||||||
<KillsCounter
|
<KillsCounter
|
||||||
killsCount={localKillsCount}
|
killsCount={localKillsCount}
|
||||||
systemId={nodeVars.solarSystemId}
|
systemId={nodeVars.solarSystemId}
|
||||||
size={TooltipSize.lg}
|
size={TooltipSize.lg}
|
||||||
killsActivityType={nodeVars.killsActivityType}
|
killsActivityType={localKillsActivityType}
|
||||||
className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[nodeVars.killsActivityType!])}
|
className={clsx(classes.Bookmark, MARKER_BOOKMARK_BG_STYLES[localKillsActivityType])}
|
||||||
>
|
>
|
||||||
<div className={clsx(classes.BookmarkWithIcon)}>
|
<div className={clsx(classes.BookmarkWithIcon)}>
|
||||||
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
<span className={clsx(PrimeIcons.BOLT, classes.icon)} />
|
||||||
<span className={clsx(classes.text)}>{nodeVars.killsCount}</span>
|
<span className={clsx(classes.text)}>{localKillsCount}</span>
|
||||||
</div>
|
</div>
|
||||||
</KillsCounter>
|
</KillsCounter>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ export function useKillsCounter({ realSystemId }: UseKillsCounterProps) {
|
|||||||
systemId: realSystemId,
|
systemId: realSystemId,
|
||||||
outCommand,
|
outCommand,
|
||||||
showAllVisible: false,
|
showAllVisible: false,
|
||||||
|
sinceHours: 1,
|
||||||
});
|
});
|
||||||
|
|
||||||
const filteredKills = useMemo(() => {
|
const filteredKills = useMemo(() => {
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import { useEffect, useState, useCallback } from 'react';
|
import { useEffect, useState, useCallback, useMemo } from 'react';
|
||||||
import { useMapEventListener } from '@/hooks/Mapper/events';
|
import { useMapEventListener } from '@/hooks/Mapper/events';
|
||||||
import { Commands } from '@/hooks/Mapper/types';
|
import { Commands } from '@/hooks/Mapper/types';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
interface Kill {
|
interface Kill {
|
||||||
solar_system_id: number | string;
|
solar_system_id: number | string;
|
||||||
@@ -9,32 +10,78 @@ interface Kill {
|
|||||||
|
|
||||||
interface MapEvent {
|
interface MapEvent {
|
||||||
name: Commands;
|
name: Commands;
|
||||||
data?: any;
|
data?: unknown;
|
||||||
payload?: Kill[];
|
payload?: Kill[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useNodeKillsCount(systemId: number | string, initialKillsCount: number | null): number | null {
|
function getActivityType(count: number): string {
|
||||||
|
if (count <= 5) return 'activityNormal';
|
||||||
|
if (count <= 30) return 'activityWarn';
|
||||||
|
return 'activityDanger';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function useNodeKillsCount(systemId: number | string, initialKillsCount: number | null = null): { killsCount: number | null; killsActivityType: string | null } {
|
||||||
const [killsCount, setKillsCount] = useState<number | null>(initialKillsCount);
|
const [killsCount, setKillsCount] = useState<number | null>(initialKillsCount);
|
||||||
|
const { data: mapData } = useMapRootState();
|
||||||
|
const { detailedKills = {} } = mapData;
|
||||||
|
|
||||||
|
// Calculate 1-hour kill count from detailed kills
|
||||||
|
const oneHourKillCount = useMemo(() => {
|
||||||
|
const systemKills = detailedKills[systemId] || [];
|
||||||
|
|
||||||
|
// If we have detailed kills data (even if empty), use it for counting
|
||||||
|
if (Object.prototype.hasOwnProperty.call(detailedKills, systemId)) {
|
||||||
|
const oneHourAgo = Date.now() - 60 * 60 * 1000; // 1 hour in milliseconds
|
||||||
|
const recentKills = systemKills.filter(kill => {
|
||||||
|
if (!kill.kill_time) return false;
|
||||||
|
const killTime = new Date(kill.kill_time).getTime();
|
||||||
|
if (isNaN(killTime)) return false;
|
||||||
|
return killTime >= oneHourAgo;
|
||||||
|
});
|
||||||
|
|
||||||
|
return recentKills.length; // Return 0 if no recent kills, not null
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return null only if we don't have detailed kills data for this system
|
||||||
|
return null;
|
||||||
|
}, [detailedKills, systemId]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
// Always prefer the calculated 1-hour count over initial count
|
||||||
|
// This ensures we properly expire old kills
|
||||||
|
if (oneHourKillCount !== null) {
|
||||||
|
setKillsCount(oneHourKillCount);
|
||||||
|
} else if (detailedKills[systemId] && detailedKills[systemId].length === 0) {
|
||||||
|
// If we have detailed kills data but it's empty, set to 0
|
||||||
|
setKillsCount(0);
|
||||||
|
} else {
|
||||||
|
// Only fall back to initial count if we have no detailed kills data at all
|
||||||
setKillsCount(initialKillsCount);
|
setKillsCount(initialKillsCount);
|
||||||
}, [initialKillsCount]);
|
}
|
||||||
|
}, [oneHourKillCount, initialKillsCount, detailedKills, systemId]);
|
||||||
|
|
||||||
const handleEvent = useCallback(
|
const handleEvent = useCallback(
|
||||||
(event: MapEvent): boolean => {
|
(event: MapEvent): boolean => {
|
||||||
if (event.name === Commands.killsUpdated && Array.isArray(event.payload)) {
|
if (event.name === Commands.killsUpdated && Array.isArray(event.payload)) {
|
||||||
const killForSystem = event.payload.find(kill => kill.solar_system_id.toString() === systemId.toString());
|
const killForSystem = event.payload.find(kill => kill.solar_system_id.toString() === systemId.toString());
|
||||||
if (killForSystem && typeof killForSystem.kills === 'number') {
|
if (killForSystem && typeof killForSystem.kills === 'number') {
|
||||||
|
// Only update if we don't have detailed kills data
|
||||||
|
if (!detailedKills[systemId] || detailedKills[systemId].length === 0) {
|
||||||
setKillsCount(killForSystem.kills);
|
setKillsCount(killForSystem.kills);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
},
|
},
|
||||||
[systemId],
|
[systemId, detailedKills],
|
||||||
);
|
);
|
||||||
|
|
||||||
useMapEventListener(handleEvent);
|
useMapEventListener(handleEvent);
|
||||||
|
|
||||||
return killsCount;
|
const killsActivityType = useMemo(() => {
|
||||||
|
return killsCount !== null && killsCount > 0 ? getActivityType(killsCount) : null;
|
||||||
|
}, [killsCount]);
|
||||||
|
|
||||||
|
return { killsCount, killsActivityType };
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
|||||||
import { useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
import { useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider';
|
import { useMapState } from '@/hooks/Mapper/components/map/MapProvider';
|
||||||
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick';
|
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick';
|
||||||
import { REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
|
import { Regions, REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
|
||||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace';
|
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace';
|
||||||
import { getSystemClassStyles } from '@/hooks/Mapper/components/map/helpers';
|
import { getSystemClassStyles } from '@/hooks/Mapper/components/map/helpers';
|
||||||
import { sortWHClasses } from '@/hooks/Mapper/helpers';
|
import { sortWHClasses } from '@/hooks/Mapper/helpers';
|
||||||
@@ -15,20 +15,12 @@ import { useSystemName } from './useSystemName';
|
|||||||
import { LabelInfo, useLabelsInfo } from './useLabelsInfo';
|
import { LabelInfo, useLabelsInfo } from './useLabelsInfo';
|
||||||
import { getSystemStaticInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
|
import { getSystemStaticInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useLoadSystemStatic';
|
||||||
|
|
||||||
function getActivityType(count: number): string {
|
|
||||||
if (count <= 5) return 'activityNormal';
|
|
||||||
if (count <= 30) return 'activityWarn';
|
|
||||||
return 'activityDanger';
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface SolarSystemNodeVars {
|
export interface SolarSystemNodeVars {
|
||||||
id: string;
|
id: string;
|
||||||
selected: boolean;
|
selected: boolean;
|
||||||
visible: boolean;
|
visible: boolean;
|
||||||
isWormhole: boolean;
|
isWormhole: boolean;
|
||||||
classTitleColor: string | null;
|
classTitleColor: string | null;
|
||||||
killsCount: number | null;
|
|
||||||
killsActivityType: string | null;
|
|
||||||
hasUserCharacters: boolean;
|
hasUserCharacters: boolean;
|
||||||
showHandlers: boolean;
|
showHandlers: boolean;
|
||||||
regionClass: string | null;
|
regionClass: string | null;
|
||||||
@@ -65,6 +57,7 @@ const SpaceToClass: Record<string, string> = {
|
|||||||
[Spaces.Matar]: 'Mataria',
|
[Spaces.Matar]: 'Mataria',
|
||||||
[Spaces.Amarr]: 'Amarria',
|
[Spaces.Amarr]: 'Amarria',
|
||||||
[Spaces.Gallente]: 'Gallente',
|
[Spaces.Gallente]: 'Gallente',
|
||||||
|
[Spaces.Pochven]: 'Pochven',
|
||||||
};
|
};
|
||||||
|
|
||||||
export function useLocalCounter(nodeVars: SolarSystemNodeVars) {
|
export function useLocalCounter(nodeVars: SolarSystemNodeVars) {
|
||||||
@@ -112,6 +105,7 @@ export const useSolarSystemNode = (props: NodeProps<MapSolarSystemType>): SolarS
|
|||||||
region_id,
|
region_id,
|
||||||
is_shattered,
|
is_shattered,
|
||||||
solar_system_name,
|
solar_system_name,
|
||||||
|
constellation_name,
|
||||||
} = systemStaticInfo;
|
} = systemStaticInfo;
|
||||||
|
|
||||||
const { isShowUnsplashedSignatures } = interfaceSettings;
|
const { isShowUnsplashedSignatures } = interfaceSettings;
|
||||||
@@ -124,7 +118,6 @@ export const useSolarSystemNode = (props: NodeProps<MapSolarSystemType>): SolarS
|
|||||||
characters,
|
characters,
|
||||||
wormholesData,
|
wormholesData,
|
||||||
hubs,
|
hubs,
|
||||||
kills,
|
|
||||||
userCharacters,
|
userCharacters,
|
||||||
isConnecting,
|
isConnecting,
|
||||||
hoverNodeId,
|
hoverNodeId,
|
||||||
@@ -161,9 +154,6 @@ export const useSolarSystemNode = (props: NodeProps<MapSolarSystemType>): SolarS
|
|||||||
isShowLinkedSigId,
|
isShowLinkedSigId,
|
||||||
});
|
});
|
||||||
|
|
||||||
const killsCount = useMemo(() => kills[parseInt(solar_system_id)] ?? null, [kills, solar_system_id]);
|
|
||||||
const killsActivityType = killsCount ? getActivityType(killsCount) : null;
|
|
||||||
|
|
||||||
const hasUserCharacters = useMemo(
|
const hasUserCharacters = useMemo(
|
||||||
() => charactersInSystem.some(x => userCharacters.includes(x.eve_id)),
|
() => charactersInSystem.some(x => userCharacters.includes(x.eve_id)),
|
||||||
[charactersInSystem, userCharacters],
|
[charactersInSystem, userCharacters],
|
||||||
@@ -195,18 +185,24 @@ export const useSolarSystemNode = (props: NodeProps<MapSolarSystemType>): SolarS
|
|||||||
const hubsAsStrings = useMemo(() => hubs.map(item => item.toString()), [hubs]);
|
const hubsAsStrings = useMemo(() => hubs.map(item => item.toString()), [hubs]);
|
||||||
|
|
||||||
const isRally = useMemo(
|
const isRally = useMemo(
|
||||||
() => pings.find(x => x.solar_system_id === solar_system_id && x.type === PingType.Rally),
|
() => !!pings.find(x => x.solar_system_id === solar_system_id && x.type === PingType.Rally),
|
||||||
[pings, solar_system_id],
|
[pings, solar_system_id],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const regionName = useMemo(() => {
|
||||||
|
if (region_id === Regions.Pochven) {
|
||||||
|
return constellation_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
return region_name;
|
||||||
|
}, [constellation_name, region_id, region_name]);
|
||||||
|
|
||||||
const nodeVars: SolarSystemNodeVars = {
|
const nodeVars: SolarSystemNodeVars = {
|
||||||
id,
|
id,
|
||||||
selected,
|
selected,
|
||||||
visible,
|
visible,
|
||||||
isWormhole,
|
isWormhole,
|
||||||
classTitleColor,
|
classTitleColor,
|
||||||
killsCount,
|
|
||||||
killsActivityType,
|
|
||||||
hasUserCharacters,
|
hasUserCharacters,
|
||||||
userCharacters,
|
userCharacters,
|
||||||
showHandlers,
|
showHandlers,
|
||||||
@@ -233,7 +229,7 @@ export const useSolarSystemNode = (props: NodeProps<MapSolarSystemType>): SolarS
|
|||||||
isThickConnections,
|
isThickConnections,
|
||||||
classTitle: class_title,
|
classTitle: class_title,
|
||||||
temporaryName: computedTemporaryName,
|
temporaryName: computedTemporaryName,
|
||||||
regionName: region_name,
|
regionName,
|
||||||
solarSystemName: solar_system_name,
|
solarSystemName: solar_system_name,
|
||||||
isRally,
|
isRally,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { useMemo } from 'react';
|
|
||||||
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
|
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
|
||||||
|
import { useMemo } from 'react';
|
||||||
|
|
||||||
interface UseSystemNameParams {
|
interface UseSystemNameParams {
|
||||||
isTempSystemNameEnabled: boolean;
|
isTempSystemNameEnabled: boolean;
|
||||||
@@ -26,7 +26,7 @@ export const useSystemName = ({
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (isShowLinkedSigIdTempName && linkedSigPrefix) {
|
if (isShowLinkedSigIdTempName && linkedSigPrefix) {
|
||||||
return temporary_name ? `${linkedSigPrefix}・${temporary_name}` : `${linkedSigPrefix}・${solar_system_name}`;
|
return temporary_name ? `${linkedSigPrefix}:${temporary_name}` : `${linkedSigPrefix}:${solar_system_name}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
return temporary_name ?? '';
|
return temporary_name ?? '';
|
||||||
|
|||||||
@@ -1,37 +1,48 @@
|
|||||||
import { Position, internalsSymbol } from 'reactflow';
|
import { Position, internalsSymbol, Node } from 'reactflow';
|
||||||
|
|
||||||
// returns the position (top,right,bottom or right) passed node compared to
|
type Coords = [number, number];
|
||||||
function getParams(nodeA, nodeB) {
|
type CoordsWithPosition = [number, number, Position];
|
||||||
|
|
||||||
|
function segmentsIntersect(a1: number, a2: number, b1: number, b2: number): boolean {
|
||||||
|
const [minA, maxA] = a1 < a2 ? [a1, a2] : [a2, a1];
|
||||||
|
const [minB, maxB] = b1 < b2 ? [b1, b2] : [b2, b1];
|
||||||
|
|
||||||
|
return maxA >= minB && maxB >= minA;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getParams(nodeA: Node, nodeB: Node): CoordsWithPosition {
|
||||||
const centerA = getNodeCenter(nodeA);
|
const centerA = getNodeCenter(nodeA);
|
||||||
const centerB = getNodeCenter(nodeB);
|
const centerB = getNodeCenter(nodeB);
|
||||||
|
|
||||||
const horizontalDiff = Math.abs(centerA.x - centerB.x);
|
|
||||||
const verticalDiff = Math.abs(centerA.y - centerB.y);
|
|
||||||
|
|
||||||
let position: Position;
|
let position: Position;
|
||||||
|
|
||||||
// when the horizontal difference between the nodes is bigger, we use Position.Left or Position.Right for the handle
|
if (
|
||||||
if (horizontalDiff > verticalDiff) {
|
segmentsIntersect(
|
||||||
position = centerA.x > centerB.x ? Position.Left : Position.Right;
|
nodeA.positionAbsolute!.x - 10,
|
||||||
} else {
|
nodeA.positionAbsolute!.x - 10 + nodeA.width! + 20,
|
||||||
// here the vertical difference between the nodes is bigger, so we use Position.Top or Position.Bottom for the handle
|
nodeB.positionAbsolute!.x,
|
||||||
|
nodeB.positionAbsolute!.x + nodeB.width!,
|
||||||
|
)
|
||||||
|
) {
|
||||||
position = centerA.y > centerB.y ? Position.Top : Position.Bottom;
|
position = centerA.y > centerB.y ? Position.Top : Position.Bottom;
|
||||||
|
} else {
|
||||||
|
position = centerA.x > centerB.x ? Position.Left : Position.Right;
|
||||||
}
|
}
|
||||||
|
|
||||||
const [x, y] = getHandleCoordsByPosition(nodeA, position);
|
const [x, y] = getHandleCoordsByPosition(nodeA, position);
|
||||||
return [x, y, position];
|
return [x, y, position];
|
||||||
}
|
}
|
||||||
|
|
||||||
function getHandleCoordsByPosition(node, handlePosition) {
|
function getHandleCoordsByPosition(node: Node, handlePosition: Position): Coords {
|
||||||
// all handles are from type source, that's why we use handleBounds.source here
|
const handle = node[internalsSymbol]!.handleBounds!.source!.find(h => h.position === handlePosition);
|
||||||
const handle = node[internalsSymbol].handleBounds.source.find(h => h.position === handlePosition);
|
|
||||||
|
if (!handle) {
|
||||||
|
throw new Error(`Handle with position ${handlePosition} not found on node ${node.id}`);
|
||||||
|
}
|
||||||
|
|
||||||
let offsetX = handle.width / 2;
|
let offsetX = handle.width / 2;
|
||||||
let offsetY = handle.height / 2;
|
let offsetY = handle.height / 2;
|
||||||
|
|
||||||
// this is a tiny detail to make the markerEnd of an edge visible.
|
|
||||||
// The handle position that gets calculated has the origin top-left, so depending which side we are using, we add a little offset
|
|
||||||
// when the handlePosition is Position.Right for example, we need to add an offset as big as the handle itself in order to get the correct position
|
|
||||||
switch (handlePosition) {
|
switch (handlePosition) {
|
||||||
case Position.Left:
|
case Position.Left:
|
||||||
offsetX = 0;
|
offsetX = 0;
|
||||||
@@ -47,21 +58,20 @@ function getHandleCoordsByPosition(node, handlePosition) {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
const x = node.positionAbsolute.x + handle.x + offsetX;
|
const x = node.positionAbsolute!.x + handle.x + offsetX;
|
||||||
const y = node.positionAbsolute.y + handle.y + offsetY;
|
const y = node.positionAbsolute!.y + handle.y + offsetY;
|
||||||
|
|
||||||
return [x, y];
|
return [x, y];
|
||||||
}
|
}
|
||||||
|
|
||||||
function getNodeCenter(node) {
|
function getNodeCenter(node: Node): { x: number; y: number } {
|
||||||
return {
|
return {
|
||||||
x: node.positionAbsolute.x + node.width / 2,
|
x: node.positionAbsolute!.x + node.width! / 2,
|
||||||
y: node.positionAbsolute.y + node.height / 2,
|
y: node.positionAbsolute!.y + node.height! / 2,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// returns the parameters (sx, sy, tx, ty, sourcePos, targetPos) you need to create an edge
|
export function getEdgeParams(source: Node, target: Node) {
|
||||||
export function getEdgeParams(source, target) {
|
|
||||||
const [sx, sy, sourcePos] = getParams(source, target);
|
const [sx, sy, sourcePos] = getParams(source, target);
|
||||||
const [tx, ty, targetPos] = getParams(target, source);
|
const [tx, ty, targetPos] = getParams(target, source);
|
||||||
|
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ import { PrimeIcons } from 'primereact/api';
|
|||||||
import { ConfirmPopup } from 'primereact/confirmpopup';
|
import { ConfirmPopup } from 'primereact/confirmpopup';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types';
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
import { useConfirmPopup } from '@/hooks/Mapper/hooks';
|
||||||
|
|
||||||
const TOOLTIP_PROPS = { content: 'Remove comment', position: TooltipPosition.top };
|
const TOOLTIP_PROPS = { content: 'Remove comment', position: TooltipPosition.top };
|
||||||
|
|
||||||
@@ -28,8 +29,7 @@ export const MarkdownComment = ({ text, time, characterEveId, id }: MarkdownComm
|
|||||||
const char = useGetCacheCharacter(characterEveId);
|
const char = useGetCacheCharacter(characterEveId);
|
||||||
const [hovered, setHovered] = useState(false);
|
const [hovered, setHovered] = useState(false);
|
||||||
|
|
||||||
const cpRemoveBtnRef = useRef<HTMLElement>();
|
const { cfShow, cfHide, cfVisible, cfRef } = useConfirmPopup();
|
||||||
const [cpRemoveVisible, setCpRemoveVisible] = useState(false);
|
|
||||||
|
|
||||||
const { outCommand } = useMapRootState();
|
const { outCommand } = useMapRootState();
|
||||||
const ref = useRef({ outCommand, id });
|
const ref = useRef({ outCommand, id });
|
||||||
@@ -45,9 +45,6 @@ export const MarkdownComment = ({ text, time, characterEveId, id }: MarkdownComm
|
|||||||
const handleMouseEnter = useCallback(() => setHovered(true), []);
|
const handleMouseEnter = useCallback(() => setHovered(true), []);
|
||||||
const handleMouseLeave = useCallback(() => setHovered(false), []);
|
const handleMouseLeave = useCallback(() => setHovered(false), []);
|
||||||
|
|
||||||
const handleShowCP = useCallback(() => setCpRemoveVisible(true), []);
|
|
||||||
const handleHideCP = useCallback(() => setCpRemoveVisible(false), []);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<InfoDrawer
|
<InfoDrawer
|
||||||
@@ -68,11 +65,11 @@ export const MarkdownComment = ({ text, time, characterEveId, id }: MarkdownComm
|
|||||||
{!hovered && <TimeAgo timestamp={time} />}
|
{!hovered && <TimeAgo timestamp={time} />}
|
||||||
{hovered && (
|
{hovered && (
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
<div ref={cpRemoveBtnRef}>
|
<div ref={cfRef}>
|
||||||
<WdImgButton
|
<WdImgButton
|
||||||
className={clsx(PrimeIcons.TRASH, 'hover:text-red-400')}
|
className={clsx(PrimeIcons.TRASH, 'hover:text-red-400')}
|
||||||
tooltip={TOOLTIP_PROPS}
|
tooltip={TOOLTIP_PROPS}
|
||||||
onClick={handleShowCP}
|
onClick={cfShow}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
@@ -85,9 +82,9 @@ export const MarkdownComment = ({ text, time, characterEveId, id }: MarkdownComm
|
|||||||
</InfoDrawer>
|
</InfoDrawer>
|
||||||
|
|
||||||
<ConfirmPopup
|
<ConfirmPopup
|
||||||
target={cpRemoveBtnRef.current}
|
target={cfRef.current}
|
||||||
visible={cpRemoveVisible}
|
visible={cfVisible}
|
||||||
onHide={handleHideCP}
|
onHide={cfHide}
|
||||||
message="Are you sure you want to delete?"
|
message="Are you sure you want to delete?"
|
||||||
icon="pi pi-exclamation-triangle"
|
icon="pi pi-exclamation-triangle"
|
||||||
accept={handleDelete}
|
accept={handleDelete}
|
||||||
|
|||||||
@@ -1,9 +1,4 @@
|
|||||||
import { Button } from 'primereact/button';
|
import { PingRoute } from '@/hooks/Mapper/components/mapInterface/components/PingsInterface/PingRoute.tsx';
|
||||||
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
|
||||||
import { Toast } from 'primereact/toast';
|
|
||||||
import clsx from 'clsx';
|
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
|
||||||
import { Commands, OutCommand, PingType } from '@/hooks/Mapper/types';
|
|
||||||
import {
|
import {
|
||||||
CharacterCardById,
|
CharacterCardById,
|
||||||
SystemView,
|
SystemView,
|
||||||
@@ -12,12 +7,18 @@ import {
|
|||||||
WdImgButton,
|
WdImgButton,
|
||||||
WdImgButtonTooltip,
|
WdImgButtonTooltip,
|
||||||
} from '@/hooks/Mapper/components/ui-kit';
|
} from '@/hooks/Mapper/components/ui-kit';
|
||||||
import useRefState from 'react-usestateref';
|
|
||||||
import { PrimeIcons } from 'primereact/api';
|
|
||||||
import { emitMapEvent } from '@/hooks/Mapper/events';
|
import { emitMapEvent } from '@/hooks/Mapper/events';
|
||||||
import { ConfirmPopup } from 'primereact/confirmpopup';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { PingRoute } from '@/hooks/Mapper/components/mapInterface/components/PingsInterface/PingRoute.tsx';
|
|
||||||
import { PingsPlacement } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
import { PingsPlacement } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import { Commands, OutCommand, PingType } from '@/hooks/Mapper/types';
|
||||||
|
import clsx from 'clsx';
|
||||||
|
import { PrimeIcons } from 'primereact/api';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { ConfirmPopup } from 'primereact/confirmpopup';
|
||||||
|
import { Toast } from 'primereact/toast';
|
||||||
|
import { useCallback, useEffect, useMemo, useRef } from 'react';
|
||||||
|
import useRefState from 'react-usestateref';
|
||||||
|
import { useConfirmPopup } from '@/hooks/Mapper/hooks';
|
||||||
|
|
||||||
const PING_PLACEMENT_MAP = {
|
const PING_PLACEMENT_MAP = {
|
||||||
[PingsPlacement.rightTop]: 'top-right',
|
[PingsPlacement.rightTop]: 'top-right',
|
||||||
@@ -78,9 +79,7 @@ export interface PingsInterfaceProps {
|
|||||||
export const PingsInterface = ({ hasLeftOffset }: PingsInterfaceProps) => {
|
export const PingsInterface = ({ hasLeftOffset }: PingsInterfaceProps) => {
|
||||||
const toast = useRef<Toast>(null);
|
const toast = useRef<Toast>(null);
|
||||||
const [isShow, setIsShow, isShowRef] = useRefState(false);
|
const [isShow, setIsShow, isShowRef] = useRefState(false);
|
||||||
|
const { cfShow, cfHide, cfVisible, cfRef } = useConfirmPopup();
|
||||||
const cpRemoveBtnRef = useRef<HTMLElement>();
|
|
||||||
const [cpRemoveVisible, setCpRemoveVisible] = useState(false);
|
|
||||||
|
|
||||||
const {
|
const {
|
||||||
storedSettings: { interfaceSettings },
|
storedSettings: { interfaceSettings },
|
||||||
@@ -98,9 +97,6 @@ export const PingsInterface = ({ hasLeftOffset }: PingsInterfaceProps) => {
|
|||||||
|
|
||||||
const ping = useMemo(() => (pings.length === 1 ? pings[0] : null), [pings]);
|
const ping = useMemo(() => (pings.length === 1 ? pings[0] : null), [pings]);
|
||||||
|
|
||||||
const handleShowCP = useCallback(() => setCpRemoveVisible(true), []);
|
|
||||||
const handleHideCP = useCallback(() => setCpRemoveVisible(false), []);
|
|
||||||
|
|
||||||
const navigateTo = useCallback(() => {
|
const navigateTo = useCallback(() => {
|
||||||
if (!ping) {
|
if (!ping) {
|
||||||
return;
|
return;
|
||||||
@@ -119,7 +115,7 @@ export const PingsInterface = ({ hasLeftOffset }: PingsInterfaceProps) => {
|
|||||||
|
|
||||||
await outCommand({
|
await outCommand({
|
||||||
type: OutCommand.cancelPing,
|
type: OutCommand.cancelPing,
|
||||||
data: { type: ping.type, solar_system_id: ping.solar_system_id },
|
data: { type: ping.type, id: ping.id },
|
||||||
});
|
});
|
||||||
}, [outCommand, ping]);
|
}, [outCommand, ping]);
|
||||||
|
|
||||||
@@ -242,11 +238,11 @@ export const PingsInterface = ({ hasLeftOffset }: PingsInterfaceProps) => {
|
|||||||
/>
|
/>
|
||||||
|
|
||||||
{/*@ts-ignore*/}
|
{/*@ts-ignore*/}
|
||||||
<div ref={cpRemoveBtnRef}>
|
<div ref={cfRef}>
|
||||||
<WdImgButton
|
<WdImgButton
|
||||||
className={clsx('pi-trash', 'text-red-400 hover:text-red-300')}
|
className={clsx('pi-trash', 'text-red-400 hover:text-red-300')}
|
||||||
tooltip={DELETE_TOOLTIP_PROPS}
|
tooltip={DELETE_TOOLTIP_PROPS}
|
||||||
onClick={handleShowCP}
|
onClick={cfShow}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
{/* TODO ADD solar system menu*/}
|
{/* TODO ADD solar system menu*/}
|
||||||
@@ -272,9 +268,9 @@ export const PingsInterface = ({ hasLeftOffset }: PingsInterfaceProps) => {
|
|||||||
/>
|
/>
|
||||||
|
|
||||||
<ConfirmPopup
|
<ConfirmPopup
|
||||||
target={cpRemoveBtnRef.current}
|
target={cfRef.current}
|
||||||
visible={cpRemoveVisible}
|
visible={cfVisible}
|
||||||
onHide={handleHideCP}
|
onHide={cfHide}
|
||||||
message="Are you sure you want to delete ping?"
|
message="Are you sure you want to delete ping?"
|
||||||
icon="pi pi-exclamation-triangle text-orange-400"
|
icon="pi pi-exclamation-triangle text-orange-400"
|
||||||
accept={removePing}
|
accept={removePing}
|
||||||
|
|||||||
@@ -7,10 +7,6 @@ import {
|
|||||||
SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS,
|
SOLAR_SYSTEM_CLASSES_TO_CLASS_GROUPS,
|
||||||
WORMHOLES_ADDITIONAL_INFO_BY_SHORT_NAME,
|
WORMHOLES_ADDITIONAL_INFO_BY_SHORT_NAME,
|
||||||
} from '@/hooks/Mapper/components/map/constants.ts';
|
} from '@/hooks/Mapper/components/map/constants.ts';
|
||||||
import {
|
|
||||||
SETTINGS_KEYS,
|
|
||||||
SignatureSettingsType,
|
|
||||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
|
||||||
import { SystemSignaturesContent } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SystemSignaturesContent';
|
import { SystemSignaturesContent } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SystemSignaturesContent';
|
||||||
import { K162_TYPES_MAP } from '@/hooks/Mapper/constants.ts';
|
import { K162_TYPES_MAP } from '@/hooks/Mapper/constants.ts';
|
||||||
import { getWhSize } from '@/hooks/Mapper/helpers/getWhSize';
|
import { getWhSize } from '@/hooks/Mapper/helpers/getWhSize';
|
||||||
@@ -18,6 +14,7 @@ import { parseSignatureCustomInfo } from '@/hooks/Mapper/helpers/parseSignatureC
|
|||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { CommandLinkSignatureToSystem, SignatureGroup, SystemSignature, TimeStatus } from '@/hooks/Mapper/types';
|
import { CommandLinkSignatureToSystem, SignatureGroup, SystemSignature, TimeStatus } from '@/hooks/Mapper/types';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
|
import { SETTINGS_KEYS, SignatureSettingsType } from '@/hooks/Mapper/constants/signatures';
|
||||||
|
|
||||||
const K162_SIGNATURE_TYPE = WORMHOLES_ADDITIONAL_INFO_BY_SHORT_NAME['K162'].shortName;
|
const K162_SIGNATURE_TYPE = WORMHOLES_ADDITIONAL_INFO_BY_SHORT_NAME['K162'].shortName;
|
||||||
|
|
||||||
|
|||||||
@@ -206,7 +206,7 @@ export const SystemSettingsDialog = ({ systemId, visible, setVisible }: SystemSe
|
|||||||
aria-describedby="temporaryName"
|
aria-describedby="temporaryName"
|
||||||
autoComplete="off"
|
autoComplete="off"
|
||||||
value={temporaryName}
|
value={temporaryName}
|
||||||
maxLength={10}
|
maxLength={12}
|
||||||
onChange={e => setTemporaryName(e.target.value)}
|
onChange={e => setTemporaryName(e.target.value)}
|
||||||
/>
|
/>
|
||||||
</IconField>
|
</IconField>
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import { useMapCheckPermissions, useMapGetOption } from '@/hooks/Mapper/mapRootP
|
|||||||
import { UserPermission } from '@/hooks/Mapper/types/permissions';
|
import { UserPermission } from '@/hooks/Mapper/types/permissions';
|
||||||
import { LocalCharactersList } from './components/LocalCharactersList';
|
import { LocalCharactersList } from './components/LocalCharactersList';
|
||||||
import { useLocalCharactersItemTemplate } from './hooks/useLocalCharacters';
|
import { useLocalCharactersItemTemplate } from './hooks/useLocalCharacters';
|
||||||
import { useLocalCharacterWidgetSettings } from './hooks/useLocalWidgetSettings';
|
|
||||||
import { LocalCharactersHeader } from './components/LocalCharactersHeader';
|
import { LocalCharactersHeader } from './components/LocalCharactersHeader';
|
||||||
import classes from './LocalCharacters.module.scss';
|
import classes from './LocalCharacters.module.scss';
|
||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
@@ -14,9 +13,9 @@ import clsx from 'clsx';
|
|||||||
export const LocalCharacters = () => {
|
export const LocalCharacters = () => {
|
||||||
const {
|
const {
|
||||||
data: { characters, userCharacters, selectedSystems },
|
data: { characters, userCharacters, selectedSystems },
|
||||||
|
storedSettings: { settingsLocal, settingsLocalUpdate },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const [settings, setSettings] = useLocalCharacterWidgetSettings();
|
|
||||||
const [systemId] = selectedSystems;
|
const [systemId] = selectedSystems;
|
||||||
const restrictOfflineShowing = useMapGetOption('restrict_offline_showing');
|
const restrictOfflineShowing = useMapGetOption('restrict_offline_showing');
|
||||||
const isAdminOrManager = useMapCheckPermissions([UserPermission.MANAGE_MAP]);
|
const isAdminOrManager = useMapCheckPermissions([UserPermission.MANAGE_MAP]);
|
||||||
@@ -31,12 +30,12 @@ export const LocalCharacters = () => {
|
|||||||
.map(x => ({
|
.map(x => ({
|
||||||
...x,
|
...x,
|
||||||
isOwn: userCharacters.includes(x.eve_id),
|
isOwn: userCharacters.includes(x.eve_id),
|
||||||
compact: settings.compact,
|
compact: settingsLocal.compact,
|
||||||
showShipName: settings.showShipName,
|
showShipName: settingsLocal.showShipName,
|
||||||
}))
|
}))
|
||||||
.sort(sortCharacters);
|
.sort(sortCharacters);
|
||||||
|
|
||||||
if (!showOffline || !settings.showOffline) {
|
if (!showOffline || !settingsLocal.showOffline) {
|
||||||
return filtered.filter(c => c.online);
|
return filtered.filter(c => c.online);
|
||||||
}
|
}
|
||||||
return filtered;
|
return filtered;
|
||||||
@@ -44,9 +43,9 @@ export const LocalCharacters = () => {
|
|||||||
characters,
|
characters,
|
||||||
systemId,
|
systemId,
|
||||||
userCharacters,
|
userCharacters,
|
||||||
settings.compact,
|
settingsLocal.compact,
|
||||||
settings.showOffline,
|
settingsLocal.showOffline,
|
||||||
settings.showShipName,
|
settingsLocal.showShipName,
|
||||||
showOffline,
|
showOffline,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@@ -54,7 +53,7 @@ export const LocalCharacters = () => {
|
|||||||
const isNotSelectedSystem = selectedSystems.length !== 1;
|
const isNotSelectedSystem = selectedSystems.length !== 1;
|
||||||
const showList = sorted.length > 0 && selectedSystems.length === 1;
|
const showList = sorted.length > 0 && selectedSystems.length === 1;
|
||||||
|
|
||||||
const itemTemplate = useLocalCharactersItemTemplate(settings.showShipName);
|
const itemTemplate = useLocalCharactersItemTemplate(settingsLocal.showShipName);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Widget
|
<Widget
|
||||||
@@ -63,8 +62,8 @@ export const LocalCharacters = () => {
|
|||||||
sortedCount={sorted.length}
|
sortedCount={sorted.length}
|
||||||
showList={showList}
|
showList={showList}
|
||||||
showOffline={showOffline}
|
showOffline={showOffline}
|
||||||
settings={settings}
|
settings={settingsLocal}
|
||||||
setSettings={setSettings}
|
setSettings={settingsLocalUpdate}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
@@ -81,7 +80,7 @@ export const LocalCharacters = () => {
|
|||||||
{showList && (
|
{showList && (
|
||||||
<LocalCharactersList
|
<LocalCharactersList
|
||||||
items={sorted}
|
items={sorted}
|
||||||
itemSize={settings.compact ? 26 : 41}
|
itemSize={settingsLocal.compact ? 26 : 41}
|
||||||
itemTemplate={itemTemplate}
|
itemTemplate={itemTemplate}
|
||||||
containerClassName={clsx(
|
containerClassName={clsx(
|
||||||
'w-full h-full overflow-x-hidden overflow-y-auto custom-scrollbar select-none',
|
'w-full h-full overflow-x-hidden overflow-y-auto custom-scrollbar select-none',
|
||||||
|
|||||||
@@ -1,21 +0,0 @@
|
|||||||
import useLocalStorageState from 'use-local-storage-state';
|
|
||||||
|
|
||||||
export interface LocalCharacterWidgetSettings {
|
|
||||||
compact: boolean;
|
|
||||||
showOffline: boolean;
|
|
||||||
version: number;
|
|
||||||
showShipName: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const LOCAL_CHARACTER_WIDGET_DEFAULT: LocalCharacterWidgetSettings = {
|
|
||||||
compact: true,
|
|
||||||
showOffline: false,
|
|
||||||
version: 0,
|
|
||||||
showShipName: false,
|
|
||||||
};
|
|
||||||
|
|
||||||
export function useLocalCharacterWidgetSettings() {
|
|
||||||
return useLocalStorageState<LocalCharacterWidgetSettings>('kills:widget:settings', {
|
|
||||||
defaultValue: LOCAL_CHARACTER_WIDGET_DEFAULT,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
@@ -8,8 +8,8 @@ import {
|
|||||||
Setting,
|
Setting,
|
||||||
SettingsTypes,
|
SettingsTypes,
|
||||||
SIGNATURE_SETTINGS,
|
SIGNATURE_SETTINGS,
|
||||||
SignatureSettingsType,
|
|
||||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
||||||
|
import { SignatureSettingsType } from '@/hooks/Mapper/constants/signatures.ts';
|
||||||
|
|
||||||
interface SystemSignatureSettingsDialogProps {
|
interface SystemSignatureSettingsDialogProps {
|
||||||
settings: SignatureSettingsType;
|
settings: SignatureSettingsType;
|
||||||
|
|||||||
@@ -1,21 +1,14 @@
|
|||||||
import { useCallback, useState, useEffect, useRef, useMemo } from 'react';
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||||
import { Widget } from '@/hooks/Mapper/components/mapInterface/components';
|
import { Widget } from '@/hooks/Mapper/components/mapInterface/components';
|
||||||
import { SystemSignaturesContent } from './SystemSignaturesContent';
|
import { SystemSignaturesContent } from './SystemSignaturesContent';
|
||||||
import { SystemSignatureSettingsDialog } from './SystemSignatureSettingsDialog';
|
import { SystemSignatureSettingsDialog } from './SystemSignatureSettingsDialog';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { SystemSignaturesHeader } from './SystemSignatureHeader';
|
import { SystemSignaturesHeader } from './SystemSignatureHeader';
|
||||||
import useLocalStorageState from 'use-local-storage-state';
|
|
||||||
import { useHotkey } from '@/hooks/Mapper/hooks/useHotkey';
|
import { useHotkey } from '@/hooks/Mapper/hooks/useHotkey';
|
||||||
import {
|
import { getDeletionTimeoutMs } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
||||||
SETTINGS_KEYS,
|
|
||||||
SETTINGS_VALUES,
|
|
||||||
SIGNATURE_SETTING_STORE_KEY,
|
|
||||||
SIGNATURE_WINDOW_ID,
|
|
||||||
SignatureSettingsType,
|
|
||||||
getDeletionTimeoutMs,
|
|
||||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
|
||||||
import { OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers';
|
import { OutCommand, OutCommandHandler } from '@/hooks/Mapper/types/mapHandlers';
|
||||||
import { ExtendedSystemSignature } from '@/hooks/Mapper/types';
|
import { ExtendedSystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { SETTINGS_KEYS, SIGNATURE_WINDOW_ID, SignatureSettingsType } from '@/hooks/Mapper/constants/signatures';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Custom hook for managing pending signature deletions and undo countdown.
|
* Custom hook for managing pending signature deletions and undo countdown.
|
||||||
@@ -70,7 +63,10 @@ function useSignatureUndo(
|
|||||||
// determine timeout from settings
|
// determine timeout from settings
|
||||||
const timeoutMs = getDeletionTimeoutMs(settings);
|
const timeoutMs = getDeletionTimeoutMs(settings);
|
||||||
|
|
||||||
setCountdown(Math.ceil(timeoutMs / 1000));
|
// Ensure a minimum of 1 second for immediate deletion so the UI shows
|
||||||
|
const effectiveTimeoutMs = timeoutMs === 0 ? 1000 : timeoutMs;
|
||||||
|
|
||||||
|
setCountdown(Math.ceil(effectiveTimeoutMs / 1000));
|
||||||
|
|
||||||
// start new interval
|
// start new interval
|
||||||
intervalRef.current = window.setInterval(() => {
|
intervalRef.current = window.setInterval(() => {
|
||||||
@@ -126,20 +122,14 @@ export const SystemSignatures = () => {
|
|||||||
const {
|
const {
|
||||||
data: { selectedSystems },
|
data: { selectedSystems },
|
||||||
outCommand,
|
outCommand,
|
||||||
|
storedSettings: { settingsSignatures, settingsSignaturesUpdate },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const [currentSettings, setCurrentSettings] = useLocalStorageState<SignatureSettingsType>(
|
|
||||||
SIGNATURE_SETTING_STORE_KEY,
|
|
||||||
{
|
|
||||||
defaultValue: SETTINGS_VALUES,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
const [systemId] = selectedSystems;
|
const [systemId] = selectedSystems;
|
||||||
const isSystemSelected = useMemo(() => selectedSystems.length === 1, [selectedSystems.length]);
|
const isSystemSelected = useMemo(() => selectedSystems.length === 1, [selectedSystems.length]);
|
||||||
const { pendingIds, countdown, deletedSignatures, addDeleted, handleUndo } = useSignatureUndo(
|
const { pendingIds, countdown, deletedSignatures, addDeleted, handleUndo } = useSignatureUndo(
|
||||||
systemId,
|
systemId,
|
||||||
currentSettings,
|
settingsSignatures,
|
||||||
outCommand,
|
outCommand,
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -157,20 +147,20 @@ export const SystemSignatures = () => {
|
|||||||
|
|
||||||
const handleSettingsSave = useCallback(
|
const handleSettingsSave = useCallback(
|
||||||
(newSettings: SignatureSettingsType) => {
|
(newSettings: SignatureSettingsType) => {
|
||||||
setCurrentSettings(newSettings);
|
settingsSignaturesUpdate(newSettings);
|
||||||
setVisible(false);
|
setVisible(false);
|
||||||
},
|
},
|
||||||
[setCurrentSettings],
|
[settingsSignaturesUpdate],
|
||||||
);
|
);
|
||||||
|
|
||||||
const handleLazyDeleteToggle = useCallback(
|
const handleLazyDeleteToggle = useCallback(
|
||||||
(value: boolean) => {
|
(value: boolean) => {
|
||||||
setCurrentSettings(prev => ({
|
settingsSignaturesUpdate(prev => ({
|
||||||
...prev,
|
...prev,
|
||||||
[SETTINGS_KEYS.LAZY_DELETE_SIGNATURES]: value,
|
[SETTINGS_KEYS.LAZY_DELETE_SIGNATURES]: value,
|
||||||
}));
|
}));
|
||||||
},
|
},
|
||||||
[setCurrentSettings],
|
[settingsSignaturesUpdate],
|
||||||
);
|
);
|
||||||
|
|
||||||
const openSettings = useCallback(() => setVisible(true), []);
|
const openSettings = useCallback(() => setVisible(true), []);
|
||||||
@@ -180,7 +170,7 @@ export const SystemSignatures = () => {
|
|||||||
label={
|
label={
|
||||||
<SystemSignaturesHeader
|
<SystemSignaturesHeader
|
||||||
sigCount={sigCount}
|
sigCount={sigCount}
|
||||||
lazyDeleteValue={currentSettings[SETTINGS_KEYS.LAZY_DELETE_SIGNATURES] as boolean}
|
lazyDeleteValue={settingsSignatures[SETTINGS_KEYS.LAZY_DELETE_SIGNATURES] as boolean}
|
||||||
pendingCount={pendingIds.size}
|
pendingCount={pendingIds.size}
|
||||||
undoCountdown={countdown}
|
undoCountdown={countdown}
|
||||||
onLazyDeleteChange={handleLazyDeleteToggle}
|
onLazyDeleteChange={handleLazyDeleteToggle}
|
||||||
@@ -197,7 +187,7 @@ export const SystemSignatures = () => {
|
|||||||
) : (
|
) : (
|
||||||
<SystemSignaturesContent
|
<SystemSignaturesContent
|
||||||
systemId={systemId}
|
systemId={systemId}
|
||||||
settings={currentSettings}
|
settings={settingsSignatures}
|
||||||
deletedSignatures={deletedSignatures}
|
deletedSignatures={deletedSignatures}
|
||||||
onLazyDeleteChange={handleLazyDeleteToggle}
|
onLazyDeleteChange={handleLazyDeleteToggle}
|
||||||
onCountChange={handleCountChange}
|
onCountChange={handleCountChange}
|
||||||
@@ -207,7 +197,7 @@ export const SystemSignatures = () => {
|
|||||||
|
|
||||||
{visible && (
|
{visible && (
|
||||||
<SystemSignatureSettingsDialog
|
<SystemSignatureSettingsDialog
|
||||||
settings={currentSettings}
|
settings={settingsSignatures}
|
||||||
onCancel={() => setVisible(false)}
|
onCancel={() => setVisible(false)}
|
||||||
onSave={handleSettingsSave}
|
onSave={handleSettingsSave}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ import {
|
|||||||
SortOrder,
|
SortOrder,
|
||||||
} from 'primereact/datatable';
|
} from 'primereact/datatable';
|
||||||
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||||
import useLocalStorageState from 'use-local-storage-state';
|
|
||||||
|
|
||||||
import { SignatureView } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SignatureView';
|
import { SignatureView } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/SignatureView';
|
||||||
import {
|
import {
|
||||||
@@ -17,9 +16,6 @@ import {
|
|||||||
GROUPS_LIST,
|
GROUPS_LIST,
|
||||||
MEDIUM_MAX_WIDTH,
|
MEDIUM_MAX_WIDTH,
|
||||||
OTHER_COLUMNS_WIDTH,
|
OTHER_COLUMNS_WIDTH,
|
||||||
SETTINGS_KEYS,
|
|
||||||
SIGNATURE_WINDOW_ID,
|
|
||||||
SignatureSettingsType,
|
|
||||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants';
|
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants';
|
||||||
import { SignatureSettings } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings';
|
import { SignatureSettings } from '@/hooks/Mapper/components/mapRootContent/components/SignatureSettings';
|
||||||
import { TooltipPosition, WdTooltip, WdTooltipHandlers, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
import { TooltipPosition, WdTooltip, WdTooltipHandlers, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
||||||
@@ -36,19 +32,11 @@ import { useClipboard, useHotkey } from '@/hooks/Mapper/hooks';
|
|||||||
import useMaxWidth from '@/hooks/Mapper/hooks/useMaxWidth';
|
import useMaxWidth from '@/hooks/Mapper/hooks/useMaxWidth';
|
||||||
import { getSignatureRowClass } from '../helpers/rowStyles';
|
import { getSignatureRowClass } from '../helpers/rowStyles';
|
||||||
import { useSystemSignaturesData } from '../hooks/useSystemSignaturesData';
|
import { useSystemSignaturesData } from '../hooks/useSystemSignaturesData';
|
||||||
|
import { SETTINGS_KEYS, SIGNATURE_WINDOW_ID, SignatureSettingsType } from '@/hooks/Mapper/constants/signatures.ts';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
const renderColIcon = (sig: SystemSignature) => renderIcon(sig);
|
const renderColIcon = (sig: SystemSignature) => renderIcon(sig);
|
||||||
|
|
||||||
type SystemSignaturesSortSettings = {
|
|
||||||
sortField: string;
|
|
||||||
sortOrder: SortOrder;
|
|
||||||
};
|
|
||||||
|
|
||||||
const SORT_DEFAULT_VALUES: SystemSignaturesSortSettings = {
|
|
||||||
sortField: 'inserted_at',
|
|
||||||
sortOrder: -1,
|
|
||||||
};
|
|
||||||
|
|
||||||
interface SystemSignaturesContentProps {
|
interface SystemSignaturesContentProps {
|
||||||
systemId: string;
|
systemId: string;
|
||||||
settings: SignatureSettingsType;
|
settings: SignatureSettingsType;
|
||||||
@@ -79,6 +67,10 @@ export const SystemSignaturesContent = ({
|
|||||||
const [nameColumnWidth, setNameColumnWidth] = useState('auto');
|
const [nameColumnWidth, setNameColumnWidth] = useState('auto');
|
||||||
const [hoveredSignature, setHoveredSignature] = useState<SystemSignature | null>(null);
|
const [hoveredSignature, setHoveredSignature] = useState<SystemSignature | null>(null);
|
||||||
|
|
||||||
|
const {
|
||||||
|
storedSettings: { settingsSignatures, settingsSignaturesUpdate },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
const tableRef = useRef<HTMLDivElement>(null);
|
const tableRef = useRef<HTMLDivElement>(null);
|
||||||
const tooltipRef = useRef<WdTooltipHandlers>(null);
|
const tooltipRef = useRef<WdTooltipHandlers>(null);
|
||||||
|
|
||||||
@@ -87,11 +79,6 @@ export const SystemSignaturesContent = ({
|
|||||||
|
|
||||||
const { clipboardContent, setClipboardContent } = useClipboard();
|
const { clipboardContent, setClipboardContent } = useClipboard();
|
||||||
|
|
||||||
const [sortSettings, setSortSettings] = useLocalStorageState<{ sortField: string; sortOrder: SortOrder }>(
|
|
||||||
'window:signatures:sort',
|
|
||||||
{ defaultValue: SORT_DEFAULT_VALUES },
|
|
||||||
);
|
|
||||||
|
|
||||||
const {
|
const {
|
||||||
signatures,
|
signatures,
|
||||||
selectedSignatures,
|
selectedSignatures,
|
||||||
@@ -246,8 +233,8 @@ export const SystemSignaturesContent = ({
|
|||||||
tooltipRef.current?.hide();
|
tooltipRef.current?.hide();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const refVars = useRef({ settings, selectedSignatures, setSortSettings });
|
const refVars = useRef({ settings, selectedSignatures, settingsSignatures, settingsSignaturesUpdate });
|
||||||
refVars.current = { settings, selectedSignatures, setSortSettings };
|
refVars.current = { settings, selectedSignatures, settingsSignatures, settingsSignaturesUpdate };
|
||||||
|
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
const getRowClassName = useCallback(rowData => {
|
const getRowClassName = useCallback(rowData => {
|
||||||
@@ -263,7 +250,12 @@ export const SystemSignaturesContent = ({
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleSortSettings = useCallback(
|
const handleSortSettings = useCallback(
|
||||||
(e: DataTableStateEvent) => refVars.current.setSortSettings({ sortField: e.sortField, sortOrder: e.sortOrder }),
|
(e: DataTableStateEvent) =>
|
||||||
|
refVars.current.settingsSignaturesUpdate({
|
||||||
|
...refVars.current.settingsSignatures,
|
||||||
|
[SETTINGS_KEYS.SORT_FIELD]: e.sortField,
|
||||||
|
[SETTINGS_KEYS.SORT_ORDER]: e.sortOrder,
|
||||||
|
}),
|
||||||
[],
|
[],
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -295,8 +287,8 @@ export const SystemSignaturesContent = ({
|
|||||||
rowHover
|
rowHover
|
||||||
selectAll
|
selectAll
|
||||||
onRowDoubleClick={handleRowClick}
|
onRowDoubleClick={handleRowClick}
|
||||||
sortField={sortSettings.sortField}
|
sortField={settingsSignatures[SETTINGS_KEYS.SORT_FIELD] as string}
|
||||||
sortOrder={sortSettings.sortOrder}
|
sortOrder={settingsSignatures[SETTINGS_KEYS.SORT_ORDER] as SortOrder}
|
||||||
onSort={handleSortSettings}
|
onSort={handleSortSettings}
|
||||||
onRowMouseEnter={onRowMouseEnter}
|
onRowMouseEnter={onRowMouseEnter}
|
||||||
onRowMouseLeave={onRowMouseLeave}
|
onRowMouseLeave={onRowMouseLeave}
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import {
|
|||||||
SignatureKindFR,
|
SignatureKindFR,
|
||||||
SignatureKindRU,
|
SignatureKindRU,
|
||||||
} from '@/hooks/Mapper/types';
|
} from '@/hooks/Mapper/types';
|
||||||
|
import { SETTINGS_KEYS, SIGNATURES_DELETION_TIMING, SignatureSettingsType } from '@/hooks/Mapper/constants/signatures';
|
||||||
|
|
||||||
export const TIME_ONE_MINUTE = 1000 * 60;
|
export const TIME_ONE_MINUTE = 1000 * 60;
|
||||||
export const TIME_TEN_MINUTES = TIME_ONE_MINUTE * 10;
|
export const TIME_TEN_MINUTES = TIME_ONE_MINUTE * 10;
|
||||||
@@ -96,44 +97,11 @@ export const getGroupIdByRawGroup = (val: string): SignatureGroup | undefined =>
|
|||||||
return MAPPING_GROUP_TO_ENG[val] || undefined;
|
return MAPPING_GROUP_TO_ENG[val] || undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const SIGNATURE_WINDOW_ID = 'system_signatures_window';
|
|
||||||
export const SIGNATURE_SETTING_STORE_KEY = 'wanderer_system_signature_settings_v6_5';
|
|
||||||
|
|
||||||
export enum SETTINGS_KEYS {
|
|
||||||
SHOW_DESCRIPTION_COLUMN = 'show_description_column',
|
|
||||||
SHOW_UPDATED_COLUMN = 'show_updated_column',
|
|
||||||
SHOW_CHARACTER_COLUMN = 'show_character_column',
|
|
||||||
LAZY_DELETE_SIGNATURES = 'lazy_delete_signatures',
|
|
||||||
KEEP_LAZY_DELETE = 'keep_lazy_delete_enabled',
|
|
||||||
DELETION_TIMING = 'deletion_timing',
|
|
||||||
COLOR_BY_TYPE = 'color_by_type',
|
|
||||||
SHOW_CHARACTER_PORTRAIT = 'show_character_portrait',
|
|
||||||
|
|
||||||
// From SignatureKind
|
|
||||||
COSMIC_ANOMALY = SignatureKind.CosmicAnomaly,
|
|
||||||
COSMIC_SIGNATURE = SignatureKind.CosmicSignature,
|
|
||||||
DEPLOYABLE = SignatureKind.Deployable,
|
|
||||||
STRUCTURE = SignatureKind.Structure,
|
|
||||||
STARBASE = SignatureKind.Starbase,
|
|
||||||
SHIP = SignatureKind.Ship,
|
|
||||||
DRONE = SignatureKind.Drone,
|
|
||||||
|
|
||||||
// From SignatureGroup
|
|
||||||
WORMHOLE = SignatureGroup.Wormhole,
|
|
||||||
RELIC_SITE = SignatureGroup.RelicSite,
|
|
||||||
DATA_SITE = SignatureGroup.DataSite,
|
|
||||||
ORE_SITE = SignatureGroup.OreSite,
|
|
||||||
GAS_SITE = SignatureGroup.GasSite,
|
|
||||||
COMBAT_SITE = SignatureGroup.CombatSite,
|
|
||||||
}
|
|
||||||
|
|
||||||
export enum SettingsTypes {
|
export enum SettingsTypes {
|
||||||
flag,
|
flag,
|
||||||
dropdown,
|
dropdown,
|
||||||
}
|
}
|
||||||
|
|
||||||
export type SignatureSettingsType = { [key in SETTINGS_KEYS]?: unknown };
|
|
||||||
|
|
||||||
export type Setting = {
|
export type Setting = {
|
||||||
key: SETTINGS_KEYS;
|
key: SETTINGS_KEYS;
|
||||||
name: string;
|
name: string;
|
||||||
@@ -142,12 +110,6 @@ export type Setting = {
|
|||||||
options?: { label: string; value: number | string | boolean }[];
|
options?: { label: string; value: number | string | boolean }[];
|
||||||
};
|
};
|
||||||
|
|
||||||
export enum SIGNATURES_DELETION_TIMING {
|
|
||||||
IMMEDIATE,
|
|
||||||
DEFAULT,
|
|
||||||
EXTENDED,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Now use a stricter type: every timing key maps to a number
|
// Now use a stricter type: every timing key maps to a number
|
||||||
export type SignatureDeletionTimingType = Record<SIGNATURES_DELETION_TIMING, number>;
|
export type SignatureDeletionTimingType = Record<SIGNATURES_DELETION_TIMING, number>;
|
||||||
|
|
||||||
@@ -194,32 +156,6 @@ export const SIGNATURE_SETTINGS = {
|
|||||||
],
|
],
|
||||||
};
|
};
|
||||||
|
|
||||||
export const SETTINGS_VALUES: SignatureSettingsType = {
|
|
||||||
[SETTINGS_KEYS.SHOW_UPDATED_COLUMN]: true,
|
|
||||||
[SETTINGS_KEYS.SHOW_DESCRIPTION_COLUMN]: true,
|
|
||||||
[SETTINGS_KEYS.SHOW_CHARACTER_COLUMN]: true,
|
|
||||||
[SETTINGS_KEYS.LAZY_DELETE_SIGNATURES]: true,
|
|
||||||
[SETTINGS_KEYS.KEEP_LAZY_DELETE]: false,
|
|
||||||
[SETTINGS_KEYS.DELETION_TIMING]: SIGNATURES_DELETION_TIMING.DEFAULT,
|
|
||||||
[SETTINGS_KEYS.COLOR_BY_TYPE]: true,
|
|
||||||
[SETTINGS_KEYS.SHOW_CHARACTER_PORTRAIT]: true,
|
|
||||||
|
|
||||||
[SETTINGS_KEYS.COSMIC_ANOMALY]: true,
|
|
||||||
[SETTINGS_KEYS.COSMIC_SIGNATURE]: true,
|
|
||||||
[SETTINGS_KEYS.DEPLOYABLE]: true,
|
|
||||||
[SETTINGS_KEYS.STRUCTURE]: true,
|
|
||||||
[SETTINGS_KEYS.STARBASE]: true,
|
|
||||||
[SETTINGS_KEYS.SHIP]: true,
|
|
||||||
[SETTINGS_KEYS.DRONE]: true,
|
|
||||||
|
|
||||||
[SETTINGS_KEYS.WORMHOLE]: true,
|
|
||||||
[SETTINGS_KEYS.RELIC_SITE]: true,
|
|
||||||
[SETTINGS_KEYS.DATA_SITE]: true,
|
|
||||||
[SETTINGS_KEYS.ORE_SITE]: true,
|
|
||||||
[SETTINGS_KEYS.GAS_SITE]: true,
|
|
||||||
[SETTINGS_KEYS.COMBAT_SITE]: true,
|
|
||||||
};
|
|
||||||
|
|
||||||
// Now this map is strongly typed as “number” for each timing enum
|
// Now this map is strongly typed as “number” for each timing enum
|
||||||
export const SIGNATURE_DELETION_TIMEOUTS: SignatureDeletionTimingType = {
|
export const SIGNATURE_DELETION_TIMEOUTS: SignatureDeletionTimingType = {
|
||||||
[SIGNATURES_DELETION_TIMING.IMMEDIATE]: 0,
|
[SIGNATURES_DELETION_TIMING.IMMEDIATE]: 0,
|
||||||
|
|||||||
@@ -0,0 +1,52 @@
|
|||||||
|
import { getState } from './getState';
|
||||||
|
import { UNKNOWN_SIGNATURE_NAME } from '@/hooks/Mapper/helpers';
|
||||||
|
import { SignatureGroup, SystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
|
describe('getState', () => {
|
||||||
|
const mockSignaturesMatch: string[] = []; // This parameter is not used in the function
|
||||||
|
|
||||||
|
it('should return 0 if group is undefined', () => {
|
||||||
|
const newSig: SystemSignature = { id: '1', name: 'Test Sig', group: undefined } as SystemSignature;
|
||||||
|
expect(getState(mockSignaturesMatch, newSig)).toBe(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 0 if group is CosmicSignature', () => {
|
||||||
|
const newSig: SystemSignature = { id: '1', name: 'Test Sig', group: SignatureGroup.CosmicSignature } as SystemSignature;
|
||||||
|
expect(getState(mockSignaturesMatch, newSig)).toBe(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 1 if group is not CosmicSignature and name is undefined', () => {
|
||||||
|
const newSig: SystemSignature = { id: '1', name: undefined, group: SignatureGroup.Wormhole } as SystemSignature;
|
||||||
|
expect(getState(mockSignaturesMatch, newSig)).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 1 if group is not CosmicSignature and name is empty', () => {
|
||||||
|
const newSig: SystemSignature = { id: '1', name: '', group: SignatureGroup.Wormhole } as SystemSignature;
|
||||||
|
expect(getState(mockSignaturesMatch, newSig)).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 1 if group is not CosmicSignature and name is UNKNOWN_SIGNATURE_NAME', () => {
|
||||||
|
const newSig: SystemSignature = { id: '1', name: UNKNOWN_SIGNATURE_NAME, group: SignatureGroup.Wormhole } as SystemSignature;
|
||||||
|
expect(getState(mockSignaturesMatch, newSig)).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return 2 if group is not CosmicSignature and name is a non-empty string', () => {
|
||||||
|
const newSig: SystemSignature = { id: '1', name: 'Custom Name', group: SignatureGroup.Wormhole } as SystemSignature;
|
||||||
|
expect(getState(mockSignaturesMatch, newSig)).toBe(2);
|
||||||
|
});
|
||||||
|
|
||||||
|
// According to the current implementation, state = -1 is unreachable
|
||||||
|
// because the conditions for 0, 1, and 2 cover all possibilities for the given inputs.
|
||||||
|
// If the logic of getState were to change to make -1 possible, a test case should be added here.
|
||||||
|
// For now, we can test a scenario that should lead to one of the valid states,
|
||||||
|
// for example, if group is something other than CosmicSignature and name is valid.
|
||||||
|
it('should handle other valid signature groups correctly, leading to state 2 with a valid name', () => {
|
||||||
|
const newSig: SystemSignature = { id: '1', name: 'Combat Site', group: SignatureGroup.CombatSite } as SystemSignature;
|
||||||
|
expect(getState(mockSignaturesMatch, newSig)).toBe(2);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle other valid signature groups correctly, leading to state 1 with an empty name', () => {
|
||||||
|
const newSig: SystemSignature = { id: '1', name: '', group: SignatureGroup.DataSite } as SystemSignature;
|
||||||
|
expect(getState(mockSignaturesMatch, newSig)).toBe(1);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import { SignatureSettingsType } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
|
||||||
import { ExtendedSystemSignature } from '@/hooks/Mapper/types';
|
import { ExtendedSystemSignature } from '@/hooks/Mapper/types';
|
||||||
|
import { SignatureSettingsType } from '@/hooks/Mapper/constants/signatures.ts';
|
||||||
|
|
||||||
export interface UseSystemSignaturesDataProps {
|
export interface UseSystemSignaturesDataProps {
|
||||||
systemId: string;
|
systemId: string;
|
||||||
|
|||||||
@@ -5,15 +5,13 @@ import { OutCommand } from '@/hooks/Mapper/types/mapHandlers';
|
|||||||
import { useCallback, useEffect, useState } from 'react';
|
import { useCallback, useEffect, useState } from 'react';
|
||||||
import useRefState from 'react-usestateref';
|
import useRefState from 'react-usestateref';
|
||||||
|
|
||||||
import {
|
import { getDeletionTimeoutMs } from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
||||||
SETTINGS_KEYS,
|
|
||||||
getDeletionTimeoutMs,
|
|
||||||
} from '@/hooks/Mapper/components/mapInterface/widgets/SystemSignatures/constants.ts';
|
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { getActualSigs } from '../helpers';
|
import { getActualSigs } from '../helpers';
|
||||||
import { UseSystemSignaturesDataProps } from './types';
|
import { UseSystemSignaturesDataProps } from './types';
|
||||||
import { usePendingDeletions } from './usePendingDeletions';
|
import { usePendingDeletions } from './usePendingDeletions';
|
||||||
import { useSignatureFetching } from './useSignatureFetching';
|
import { useSignatureFetching } from './useSignatureFetching';
|
||||||
|
import { SETTINGS_KEYS } from '@/hooks/Mapper/constants/signatures.ts';
|
||||||
|
|
||||||
export const useSystemSignaturesData = ({
|
export const useSystemSignaturesData = ({
|
||||||
systemId,
|
systemId,
|
||||||
@@ -78,17 +76,12 @@ export const useSystemSignaturesData = ({
|
|||||||
if (removed.length > 0) {
|
if (removed.length > 0) {
|
||||||
await processRemovedSignatures(removed, added, updated);
|
await processRemovedSignatures(removed, added, updated);
|
||||||
|
|
||||||
// Only show pending deletions if:
|
// Show pending deletions if lazy deletion is enabled
|
||||||
// 1. Lazy deletion is enabled AND
|
// The deletion timing controls how long the countdown lasts, not whether lazy delete is active
|
||||||
// 2. Deletion timing is not immediate (> 0)
|
|
||||||
if (onSignatureDeleted && lazyDeleteValue) {
|
if (onSignatureDeleted && lazyDeleteValue) {
|
||||||
const timeoutMs = getDeletionTimeoutMs(settings);
|
|
||||||
|
|
||||||
if (timeoutMs > 0) {
|
|
||||||
onSignatureDeleted(removed);
|
onSignatureDeleted(removed);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (updated.length !== 0 || added.length !== 0) {
|
if (updated.length !== 0 || added.length !== 0) {
|
||||||
await outCommand({
|
await outCommand({
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
|||||||
import { Widget } from '@/hooks/Mapper/components/mapInterface/components';
|
import { Widget } from '@/hooks/Mapper/components/mapInterface/components';
|
||||||
import { SystemKillsList } from './SystemKillsList';
|
import { SystemKillsList } from './SystemKillsList';
|
||||||
import { KillsHeader } from './components/SystemKillsHeader';
|
import { KillsHeader } from './components/SystemKillsHeader';
|
||||||
import { useKillsWidgetSettings } from './hooks/useKillsWidgetSettings';
|
|
||||||
import { useSystemKills } from './hooks/useSystemKills';
|
import { useSystemKills } from './hooks/useSystemKills';
|
||||||
import { KillsSettingsDialog } from './components/SystemKillsSettingsDialog';
|
import { KillsSettingsDialog } from './components/SystemKillsSettingsDialog';
|
||||||
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace';
|
import { isWormholeSpace } from '@/hooks/Mapper/components/map/helpers/isWormholeSpace';
|
||||||
@@ -13,27 +12,25 @@ const SystemKillsContent = () => {
|
|||||||
const {
|
const {
|
||||||
data: { selectedSystems, isSubscriptionActive },
|
data: { selectedSystems, isSubscriptionActive },
|
||||||
outCommand,
|
outCommand,
|
||||||
|
storedSettings: { settingsKills },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const [systemId] = selectedSystems || [];
|
const [systemId] = selectedSystems || [];
|
||||||
|
|
||||||
const systemStaticInfo = getSystemStaticInfo(systemId)!;
|
const systemStaticInfo = getSystemStaticInfo(systemId)!;
|
||||||
|
|
||||||
const [settings] = useKillsWidgetSettings();
|
|
||||||
const visible = settings.showAll;
|
|
||||||
|
|
||||||
const { kills, isLoading, error } = useSystemKills({
|
const { kills, isLoading, error } = useSystemKills({
|
||||||
systemId,
|
systemId,
|
||||||
outCommand,
|
outCommand,
|
||||||
showAllVisible: visible,
|
showAllVisible: settingsKills.showAll,
|
||||||
sinceHours: settings.timeRange,
|
sinceHours: settingsKills.timeRange,
|
||||||
});
|
});
|
||||||
|
|
||||||
const isNothingSelected = !systemId && !visible;
|
const isNothingSelected = !systemId && !settingsKills.showAll;
|
||||||
const showLoading = isLoading && kills.length === 0;
|
const showLoading = isLoading && kills.length === 0;
|
||||||
|
|
||||||
const filteredKills = useMemo(() => {
|
const filteredKills = useMemo(() => {
|
||||||
if (!settings.whOnly || !visible) return kills;
|
if (!settingsKills.whOnly || !settingsKills.showAll) return kills;
|
||||||
return kills.filter(kill => {
|
return kills.filter(kill => {
|
||||||
if (!systemStaticInfo) {
|
if (!systemStaticInfo) {
|
||||||
console.warn(`System with id ${kill.solar_system_id} not found.`);
|
console.warn(`System with id ${kill.solar_system_id} not found.`);
|
||||||
@@ -41,7 +38,7 @@ const SystemKillsContent = () => {
|
|||||||
}
|
}
|
||||||
return isWormholeSpace(systemStaticInfo.system_class);
|
return isWormholeSpace(systemStaticInfo.system_class);
|
||||||
});
|
});
|
||||||
}, [kills, settings.whOnly, systemStaticInfo, visible]);
|
}, [kills, settingsKills.whOnly, systemStaticInfo, settingsKills.showAll]);
|
||||||
|
|
||||||
if (!isSubscriptionActive) {
|
if (!isSubscriptionActive) {
|
||||||
return (
|
return (
|
||||||
@@ -87,7 +84,9 @@ const SystemKillsContent = () => {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
return <SystemKillsList kills={filteredKills} onlyOneSystem={!visible} timeRange={settings.timeRange} />;
|
return (
|
||||||
|
<SystemKillsList kills={filteredKills} onlyOneSystem={!settingsKills.showAll} timeRange={settingsKills.timeRange} />
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export const WSystemKills = () => {
|
export const WSystemKills = () => {
|
||||||
|
|||||||
@@ -17,67 +17,91 @@ import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
|||||||
import { WithClassName } from '@/hooks/Mapper/types/common.ts';
|
import { WithClassName } from '@/hooks/Mapper/types/common.ts';
|
||||||
|
|
||||||
export type CompactKillRowProps = {
|
export type CompactKillRowProps = {
|
||||||
killDetails: DetailedKill;
|
killDetails?: DetailedKill | null;
|
||||||
systemName: string;
|
systemName: string;
|
||||||
onlyOneSystem: boolean;
|
onlyOneSystem: boolean;
|
||||||
} & WithClassName;
|
} & WithClassName;
|
||||||
|
|
||||||
export const KillRowDetail = ({ killDetails, systemName, onlyOneSystem, className }: CompactKillRowProps) => {
|
export const KillRowDetail = ({ killDetails, systemName, onlyOneSystem, className }: CompactKillRowProps) => {
|
||||||
const {
|
const {
|
||||||
killmail_id = 0,
|
killmail_id,
|
||||||
// Victim data
|
// Victim data
|
||||||
victim_char_name = 'Unknown Pilot',
|
victim_char_name,
|
||||||
victim_alliance_ticker = '',
|
victim_alliance_ticker,
|
||||||
victim_corp_ticker = '',
|
victim_corp_ticker,
|
||||||
victim_ship_name = 'Unknown Ship',
|
victim_ship_name,
|
||||||
victim_corp_name = '',
|
victim_corp_name,
|
||||||
victim_alliance_name = '',
|
victim_alliance_name,
|
||||||
victim_char_id = 0,
|
|
||||||
victim_corp_id = 0,
|
|
||||||
victim_alliance_id = 0,
|
|
||||||
victim_ship_type_id = 0,
|
|
||||||
// Attacker data
|
|
||||||
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;
|
|
||||||
|
|
||||||
// Define victim affiliation ticker.
|
|
||||||
const victimAffiliationTicker = victim_alliance_ticker || victim_corp_ticker || 'No Ticker';
|
|
||||||
|
|
||||||
const killValueFormatted = total_value != null && total_value > 0 ? `${formatISK(total_value)} ISK` : null;
|
|
||||||
const killTimeAgo = kill_time ? formatTimeMixed(kill_time) : '0h ago';
|
|
||||||
|
|
||||||
const attackerSubscript = getAttackerSubscript(killDetails);
|
|
||||||
|
|
||||||
const { victimCorpLogoUrl, victimAllianceLogoUrl, victimShipUrl } = buildVictimImageUrls({
|
|
||||||
victim_char_id,
|
victim_char_id,
|
||||||
victim_ship_type_id,
|
|
||||||
victim_corp_id,
|
victim_corp_id,
|
||||||
victim_alliance_id,
|
victim_alliance_id,
|
||||||
|
victim_ship_type_id,
|
||||||
|
// Attacker data
|
||||||
|
final_blow_char_id,
|
||||||
|
final_blow_char_name,
|
||||||
|
final_blow_alliance_ticker,
|
||||||
|
final_blow_alliance_name,
|
||||||
|
final_blow_alliance_id,
|
||||||
|
final_blow_corp_ticker,
|
||||||
|
final_blow_corp_id,
|
||||||
|
final_blow_corp_name,
|
||||||
|
final_blow_ship_type_id,
|
||||||
|
kill_time,
|
||||||
|
total_value,
|
||||||
|
} = killDetails || {};
|
||||||
|
|
||||||
|
// Apply fallback values using nullish coalescing to handle both null and undefined
|
||||||
|
const safeKillmailId = killmail_id ?? 0;
|
||||||
|
const safeVictimCharName = victim_char_name ?? 'Unknown Pilot';
|
||||||
|
const safeVictimAllianceTicker = victim_alliance_ticker ?? '';
|
||||||
|
const safeVictimCorpTicker = victim_corp_ticker ?? '';
|
||||||
|
const safeVictimShipName = victim_ship_name ?? 'Unknown Ship';
|
||||||
|
const safeVictimCorpName = victim_corp_name ?? '';
|
||||||
|
const safeVictimAllianceName = victim_alliance_name ?? '';
|
||||||
|
const safeVictimCharId = victim_char_id ?? 0;
|
||||||
|
const safeVictimCorpId = victim_corp_id ?? 0;
|
||||||
|
const safeVictimAllianceId = victim_alliance_id ?? 0;
|
||||||
|
const safeVictimShipTypeId = victim_ship_type_id ?? 0;
|
||||||
|
const safeFinalBlowCharId = final_blow_char_id ?? 0;
|
||||||
|
const safeFinalBlowCharName = final_blow_char_name ?? '';
|
||||||
|
const safeFinalBlowAllianceTicker = final_blow_alliance_ticker ?? '';
|
||||||
|
const safeFinalBlowAllianceName = final_blow_alliance_name ?? '';
|
||||||
|
const safeFinalBlowAllianceId = final_blow_alliance_id ?? 0;
|
||||||
|
const safeFinalBlowCorpTicker = final_blow_corp_ticker ?? '';
|
||||||
|
const safeFinalBlowCorpId = final_blow_corp_id ?? 0;
|
||||||
|
const safeFinalBlowCorpName = final_blow_corp_name ?? '';
|
||||||
|
const safeFinalBlowShipTypeId = final_blow_ship_type_id ?? 0;
|
||||||
|
const safeKillTime = kill_time ?? '';
|
||||||
|
const safeTotalValue = total_value ?? 0;
|
||||||
|
|
||||||
|
const attackerIsNpc = safeFinalBlowCharId === 0;
|
||||||
|
|
||||||
|
// Define victim affiliation ticker.
|
||||||
|
const victimAffiliationTicker = safeVictimAllianceTicker || safeVictimCorpTicker || 'No Ticker';
|
||||||
|
|
||||||
|
const killValueFormatted = safeTotalValue != null && safeTotalValue > 0 ? `${formatISK(safeTotalValue)} ISK` : null;
|
||||||
|
const killTimeAgo = safeKillTime ? formatTimeMixed(safeKillTime) : '0h ago';
|
||||||
|
|
||||||
|
const attackerSubscript = killDetails ? getAttackerSubscript(killDetails) : undefined;
|
||||||
|
|
||||||
|
const { victimCorpLogoUrl, victimAllianceLogoUrl, victimShipUrl } = buildVictimImageUrls({
|
||||||
|
victim_char_id: safeVictimCharId,
|
||||||
|
victim_ship_type_id: safeVictimShipTypeId,
|
||||||
|
victim_corp_id: safeVictimCorpId,
|
||||||
|
victim_alliance_id: safeVictimAllianceId,
|
||||||
});
|
});
|
||||||
|
|
||||||
const { attackerCorpLogoUrl, attackerAllianceLogoUrl } = buildAttackerImageUrls({
|
const { attackerCorpLogoUrl, attackerAllianceLogoUrl } = buildAttackerImageUrls({
|
||||||
final_blow_char_id,
|
final_blow_char_id: safeFinalBlowCharId,
|
||||||
final_blow_corp_id,
|
final_blow_corp_id: safeFinalBlowCorpId,
|
||||||
final_blow_alliance_id,
|
final_blow_alliance_id: safeFinalBlowAllianceId,
|
||||||
});
|
});
|
||||||
|
|
||||||
const { url: victimPrimaryLogoUrl, tooltip: victimPrimaryTooltip } = getPrimaryLogoAndTooltip(
|
const { url: victimPrimaryLogoUrl, tooltip: victimPrimaryTooltip } = getPrimaryLogoAndTooltip(
|
||||||
victimAllianceLogoUrl,
|
victimAllianceLogoUrl,
|
||||||
victimCorpLogoUrl,
|
victimCorpLogoUrl,
|
||||||
victim_alliance_name,
|
safeVictimAllianceName,
|
||||||
victim_corp_name,
|
safeVictimCorpName,
|
||||||
'Victim',
|
'Victim',
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -87,25 +111,25 @@ export const KillRowDetail = ({ killDetails, systemName, onlyOneSystem, classNam
|
|||||||
attackerIsNpc,
|
attackerIsNpc,
|
||||||
attackerAllianceLogoUrl,
|
attackerAllianceLogoUrl,
|
||||||
attackerCorpLogoUrl,
|
attackerCorpLogoUrl,
|
||||||
final_blow_alliance_name,
|
safeFinalBlowAllianceName,
|
||||||
final_blow_corp_name,
|
safeFinalBlowCorpName,
|
||||||
final_blow_ship_type_id,
|
safeFinalBlowShipTypeId,
|
||||||
),
|
),
|
||||||
[
|
[
|
||||||
attackerAllianceLogoUrl,
|
attackerAllianceLogoUrl,
|
||||||
attackerCorpLogoUrl,
|
attackerCorpLogoUrl,
|
||||||
attackerIsNpc,
|
attackerIsNpc,
|
||||||
final_blow_alliance_name,
|
safeFinalBlowAllianceName,
|
||||||
final_blow_corp_name,
|
safeFinalBlowCorpName,
|
||||||
final_blow_ship_type_id,
|
safeFinalBlowShipTypeId,
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
|
||||||
// Define attackerTicker to use the alliance ticker if available, otherwise the corp ticker.
|
// Define attackerTicker to use the alliance ticker if available, otherwise the corp ticker.
|
||||||
const attackerTicker = attackerIsNpc ? '' : final_blow_alliance_ticker || final_blow_corp_ticker || '';
|
const attackerTicker = attackerIsNpc ? '' : safeFinalBlowAllianceTicker || safeFinalBlowCorpTicker || '';
|
||||||
|
|
||||||
// For the attacker image link: if the attacker is not an NPC, link to the character page; otherwise, link to the kill page.
|
// For the attacker image link: if the attacker is not an NPC, link to the character page; otherwise, link to the kill page.
|
||||||
const attackerLink = attackerIsNpc ? zkillLink('kill', killmail_id) : zkillLink('character', final_blow_char_id);
|
const attackerLink = attackerIsNpc ? zkillLink('kill', safeKillmailId) : zkillLink('character', safeFinalBlowCharId);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@@ -121,7 +145,7 @@ export const KillRowDetail = ({ killDetails, systemName, onlyOneSystem, classNam
|
|||||||
{victimShipUrl && (
|
{victimShipUrl && (
|
||||||
<div className="relative shrink-0 w-8 h-8 overflow-hidden">
|
<div className="relative shrink-0 w-8 h-8 overflow-hidden">
|
||||||
<a
|
<a
|
||||||
href={zkillLink('kill', killmail_id)}
|
href={zkillLink('kill', safeKillmailId)}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
className="block w-full h-full"
|
className="block w-full h-full"
|
||||||
@@ -137,7 +161,7 @@ export const KillRowDetail = ({ killDetails, systemName, onlyOneSystem, classNam
|
|||||||
{victimPrimaryLogoUrl && (
|
{victimPrimaryLogoUrl && (
|
||||||
<WdTooltipWrapper content={victimPrimaryTooltip} position={TooltipPosition.top}>
|
<WdTooltipWrapper content={victimPrimaryTooltip} position={TooltipPosition.top}>
|
||||||
<a
|
<a
|
||||||
href={zkillLink('kill', killmail_id)}
|
href={zkillLink('kill', safeKillmailId)}
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noopener noreferrer"
|
rel="noopener noreferrer"
|
||||||
className="relative block shrink-0 w-8 h-8 overflow-hidden"
|
className="relative block shrink-0 w-8 h-8 overflow-hidden"
|
||||||
@@ -153,12 +177,12 @@ export const KillRowDetail = ({ killDetails, systemName, onlyOneSystem, classNam
|
|||||||
</div>
|
</div>
|
||||||
<div className="flex flex-col ml-2 flex-1 min-w-0 overflow-hidden leading-[1rem]">
|
<div className="flex flex-col ml-2 flex-1 min-w-0 overflow-hidden leading-[1rem]">
|
||||||
<div className="truncate text-stone-200">
|
<div className="truncate text-stone-200">
|
||||||
{victim_char_name}
|
{safeVictimCharName}
|
||||||
<span className="text-stone-400"> / {victimAffiliationTicker}</span>
|
<span className="text-stone-400"> / {victimAffiliationTicker}</span>
|
||||||
</div>
|
</div>
|
||||||
<div className="truncate text-stone-300 flex items-center gap-1">
|
<div className="truncate text-stone-300 flex items-center gap-1">
|
||||||
<span className="text-stone-400 overflow-hidden text-ellipsis whitespace-nowrap max-w-[140px]">
|
<span className="text-stone-400 overflow-hidden text-ellipsis whitespace-nowrap max-w-[140px]">
|
||||||
{victim_ship_name}
|
{safeVictimShipName}
|
||||||
</span>
|
</span>
|
||||||
{killValueFormatted && (
|
{killValueFormatted && (
|
||||||
<>
|
<>
|
||||||
@@ -170,9 +194,9 @@ export const KillRowDetail = ({ killDetails, systemName, onlyOneSystem, classNam
|
|||||||
</div>
|
</div>
|
||||||
<div className="flex items-center ml-auto gap-2">
|
<div className="flex items-center ml-auto gap-2">
|
||||||
<div className="flex flex-col items-end flex-1 min-w-0 overflow-hidden text-right leading-[1rem]">
|
<div className="flex flex-col items-end flex-1 min-w-0 overflow-hidden text-right leading-[1rem]">
|
||||||
{!attackerIsNpc && (final_blow_char_name || attackerTicker) && (
|
{!attackerIsNpc && (safeFinalBlowCharName || attackerTicker) && (
|
||||||
<div className="truncate text-stone-200">
|
<div className="truncate text-stone-200">
|
||||||
{final_blow_char_name}
|
{safeFinalBlowCharName}
|
||||||
{!attackerIsNpc && attackerTicker && <span className="ml-1 text-stone-400">/ {attackerTicker}</span>}
|
{!attackerIsNpc && attackerTicker && <span className="ml-1 text-stone-400">/ {attackerTicker}</span>}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -7,9 +7,9 @@ import {
|
|||||||
WdImgButton,
|
WdImgButton,
|
||||||
WdTooltipWrapper,
|
WdTooltipWrapper,
|
||||||
} from '@/hooks/Mapper/components/ui-kit';
|
} from '@/hooks/Mapper/components/ui-kit';
|
||||||
import { useKillsWidgetSettings } from '../hooks/useKillsWidgetSettings';
|
|
||||||
import { PrimeIcons } from 'primereact/api';
|
import { PrimeIcons } from 'primereact/api';
|
||||||
import useMaxWidth from '@/hooks/Mapper/hooks/useMaxWidth.ts';
|
import useMaxWidth from '@/hooks/Mapper/hooks/useMaxWidth.ts';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
interface KillsHeaderProps {
|
interface KillsHeaderProps {
|
||||||
systemId?: string;
|
systemId?: string;
|
||||||
@@ -17,11 +17,14 @@ interface KillsHeaderProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const KillsHeader: React.FC<KillsHeaderProps> = ({ systemId, onOpenSettings }) => {
|
export const KillsHeader: React.FC<KillsHeaderProps> = ({ systemId, onOpenSettings }) => {
|
||||||
const [settings, setSettings] = useKillsWidgetSettings();
|
const {
|
||||||
const { showAll } = settings;
|
storedSettings: { settingsKills, settingsKillsUpdate },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const { showAll } = settingsKills;
|
||||||
|
|
||||||
const onToggleShowAllVisible = () => {
|
const onToggleShowAllVisible = () => {
|
||||||
setSettings(prev => ({ ...prev, showAll: !prev.showAll }));
|
settingsKillsUpdate(prev => ({ ...prev, showAll: !prev.showAll }));
|
||||||
};
|
};
|
||||||
|
|
||||||
const headerRef = useRef<HTMLDivElement>(null);
|
const headerRef = useRef<HTMLDivElement>(null);
|
||||||
|
|||||||
@@ -3,12 +3,12 @@ import { Dialog } from 'primereact/dialog';
|
|||||||
import { Button } from 'primereact/button';
|
import { Button } from 'primereact/button';
|
||||||
import { WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
import { WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
||||||
import { PrimeIcons } from 'primereact/api';
|
import { PrimeIcons } from 'primereact/api';
|
||||||
import { useKillsWidgetSettings } from '../hooks/useKillsWidgetSettings';
|
|
||||||
import {
|
import {
|
||||||
AddSystemDialog,
|
AddSystemDialog,
|
||||||
SearchOnSubmitCallback,
|
SearchOnSubmitCallback,
|
||||||
} from '@/hooks/Mapper/components/mapInterface/components/AddSystemDialog';
|
} from '@/hooks/Mapper/components/mapInterface/components/AddSystemDialog';
|
||||||
import { SystemView, TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
import { SystemView, TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
interface KillsSettingsDialogProps {
|
interface KillsSettingsDialogProps {
|
||||||
visible: boolean;
|
visible: boolean;
|
||||||
@@ -16,12 +16,15 @@ interface KillsSettingsDialogProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const KillsSettingsDialog: React.FC<KillsSettingsDialogProps> = ({ visible, setVisible }) => {
|
export const KillsSettingsDialog: React.FC<KillsSettingsDialogProps> = ({ visible, setVisible }) => {
|
||||||
const [globalSettings, setGlobalSettings] = useKillsWidgetSettings();
|
const {
|
||||||
|
storedSettings: { settingsKills, settingsKillsUpdate },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
const localRef = useRef({
|
const localRef = useRef({
|
||||||
showAll: globalSettings.showAll,
|
showAll: settingsKills.showAll,
|
||||||
whOnly: globalSettings.whOnly,
|
whOnly: settingsKills.whOnly,
|
||||||
excludedSystems: globalSettings.excludedSystems || [],
|
excludedSystems: settingsKills.excludedSystems || [],
|
||||||
timeRange: globalSettings.timeRange,
|
timeRange: settingsKills.timeRange,
|
||||||
});
|
});
|
||||||
|
|
||||||
const [, forceRender] = useState(0);
|
const [, forceRender] = useState(0);
|
||||||
@@ -30,14 +33,14 @@ export const KillsSettingsDialog: React.FC<KillsSettingsDialogProps> = ({ visibl
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (visible) {
|
if (visible) {
|
||||||
localRef.current = {
|
localRef.current = {
|
||||||
showAll: globalSettings.showAll,
|
showAll: settingsKills.showAll,
|
||||||
whOnly: globalSettings.whOnly,
|
whOnly: settingsKills.whOnly,
|
||||||
excludedSystems: globalSettings.excludedSystems || [],
|
excludedSystems: settingsKills.excludedSystems || [],
|
||||||
timeRange: globalSettings.timeRange,
|
timeRange: settingsKills.timeRange,
|
||||||
};
|
};
|
||||||
forceRender(n => n + 1);
|
forceRender(n => n + 1);
|
||||||
}
|
}
|
||||||
}, [visible, globalSettings]);
|
}, [visible, settingsKills]);
|
||||||
|
|
||||||
const handleWHChange = useCallback((checked: boolean) => {
|
const handleWHChange = useCallback((checked: boolean) => {
|
||||||
localRef.current = {
|
localRef.current = {
|
||||||
@@ -75,12 +78,12 @@ export const KillsSettingsDialog: React.FC<KillsSettingsDialogProps> = ({ visibl
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleApply = useCallback(() => {
|
const handleApply = useCallback(() => {
|
||||||
setGlobalSettings(prev => ({
|
settingsKillsUpdate(prev => ({
|
||||||
...prev,
|
...prev,
|
||||||
...localRef.current,
|
...localRef.current,
|
||||||
}));
|
}));
|
||||||
setVisible(false);
|
setVisible(false);
|
||||||
}, [setGlobalSettings, setVisible]);
|
}, [settingsKillsUpdate, setVisible]);
|
||||||
|
|
||||||
const handleHide = useCallback(() => {
|
const handleHide = useCallback(() => {
|
||||||
setVisible(false);
|
setVisible(false);
|
||||||
|
|||||||
@@ -33,7 +33,10 @@ export function formatISK(value: number): string {
|
|||||||
return Math.round(value).toString();
|
return Math.round(value).toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getAttackerSubscript(kill: DetailedKill) {
|
export function getAttackerSubscript(kill: DetailedKill | undefined) {
|
||||||
|
if (!kill) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
if (kill.npc) {
|
if (kill.npc) {
|
||||||
return { label: 'npc', cssClass: 'text-purple-400' };
|
return { label: 'npc', cssClass: 'text-purple-400' };
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,53 +0,0 @@
|
|||||||
import { useMemo, useCallback } from 'react';
|
|
||||||
import useLocalStorageState from 'use-local-storage-state';
|
|
||||||
|
|
||||||
export interface KillsWidgetSettings {
|
|
||||||
showAll: boolean;
|
|
||||||
whOnly: boolean;
|
|
||||||
excludedSystems: number[];
|
|
||||||
version: number;
|
|
||||||
timeRange: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const DEFAULT_KILLS_WIDGET_SETTINGS: KillsWidgetSettings = {
|
|
||||||
showAll: false,
|
|
||||||
whOnly: true,
|
|
||||||
excludedSystems: [],
|
|
||||||
version: 2,
|
|
||||||
timeRange: 4,
|
|
||||||
};
|
|
||||||
|
|
||||||
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;
|
|
||||||
}
|
|
||||||
@@ -3,7 +3,6 @@ import debounce from 'lodash.debounce';
|
|||||||
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers';
|
import { OutCommand } from '@/hooks/Mapper/types/mapHandlers';
|
||||||
import { DetailedKill } from '@/hooks/Mapper/types/kills';
|
import { DetailedKill } from '@/hooks/Mapper/types/kills';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { useKillsWidgetSettings } from './useKillsWidgetSettings';
|
|
||||||
|
|
||||||
interface UseSystemKillsProps {
|
interface UseSystemKillsProps {
|
||||||
systemId?: string;
|
systemId?: string;
|
||||||
@@ -13,26 +12,25 @@ interface UseSystemKillsProps {
|
|||||||
sinceHours?: number;
|
sinceHours?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
function combineKills(existing: DetailedKill[], incoming: DetailedKill[], sinceHours: number): DetailedKill[] {
|
function combineKills(existing: DetailedKill[], incoming: DetailedKill[]): DetailedKill[] {
|
||||||
const cutoff = Date.now() - sinceHours * 60 * 60 * 1000;
|
// Don't filter by time when storing - let components filter when displaying
|
||||||
const byId: Record<string, DetailedKill> = {};
|
const byId: Record<string, DetailedKill> = {};
|
||||||
|
|
||||||
for (const kill of [...existing, ...incoming]) {
|
for (const kill of [...existing, ...incoming]) {
|
||||||
if (!kill.kill_time) continue;
|
if (!kill.kill_time) continue;
|
||||||
const killTimeMs = new Date(kill.kill_time).valueOf();
|
|
||||||
if (killTimeMs >= cutoff) {
|
|
||||||
byId[kill.killmail_id] = kill;
|
byId[kill.killmail_id] = kill;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return Object.values(byId);
|
return Object.values(byId);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function useSystemKills({ systemId, outCommand, showAllVisible = false, sinceHours = 24 }: UseSystemKillsProps) {
|
export function useSystemKills({ systemId, outCommand, showAllVisible = false, sinceHours = 24 }: UseSystemKillsProps) {
|
||||||
const { data, update } = useMapRootState();
|
const {
|
||||||
const { detailedKills = {}, systems = [] } = data;
|
data: { detailedKills = {}, systems = [] },
|
||||||
const [settings] = useKillsWidgetSettings();
|
update,
|
||||||
const excludedSystems = settings.excludedSystems;
|
storedSettings: { settingsKills },
|
||||||
|
} = useMapRootState();
|
||||||
|
const { excludedSystems } = settingsKills;
|
||||||
|
|
||||||
const effectiveSinceHours = sinceHours;
|
const effectiveSinceHours = sinceHours;
|
||||||
|
|
||||||
@@ -55,14 +53,14 @@ export function useSystemKills({ systemId, outCommand, showAllVisible = false, s
|
|||||||
|
|
||||||
for (const [sid, newKills] of Object.entries(killsMap)) {
|
for (const [sid, newKills] of Object.entries(killsMap)) {
|
||||||
const existing = updated[sid] ?? [];
|
const existing = updated[sid] ?? [];
|
||||||
const combined = combineKills(existing, newKills, effectiveSinceHours);
|
const combined = combineKills(existing, newKills);
|
||||||
updated[sid] = combined;
|
updated[sid] = combined;
|
||||||
}
|
}
|
||||||
|
|
||||||
return { ...prev, detailedKills: updated };
|
return { ...prev, detailedKills: updated };
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
[update, effectiveSinceHours],
|
[update],
|
||||||
);
|
);
|
||||||
|
|
||||||
const fetchKills = useCallback(
|
const fetchKills = useCallback(
|
||||||
|
|||||||
@@ -14,13 +14,14 @@ import { TrackingDialog } from '@/hooks/Mapper/components/mapRootContent/compone
|
|||||||
import { useMapEventListener } from '@/hooks/Mapper/events';
|
import { useMapEventListener } from '@/hooks/Mapper/events';
|
||||||
import { Commands } from '@/hooks/Mapper/types';
|
import { Commands } from '@/hooks/Mapper/types';
|
||||||
import { PingsInterface } from '@/hooks/Mapper/components/mapInterface/components';
|
import { PingsInterface } from '@/hooks/Mapper/components/mapInterface/components';
|
||||||
|
import { OldSettingsDialog } from '@/hooks/Mapper/components/mapRootContent/components/OldSettingsDialog.tsx';
|
||||||
|
|
||||||
export interface MapRootContentProps {}
|
export interface MapRootContentProps {}
|
||||||
|
|
||||||
// eslint-disable-next-line no-empty-pattern
|
// eslint-disable-next-line no-empty-pattern
|
||||||
export const MapRootContent = ({}: MapRootContentProps) => {
|
export const MapRootContent = ({}: MapRootContentProps) => {
|
||||||
const {
|
const {
|
||||||
storedSettings: { interfaceSettings },
|
storedSettings: { interfaceSettings, isReady, hasOldSettings },
|
||||||
data,
|
data,
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
const { isShowMenu } = interfaceSettings;
|
const { isShowMenu } = interfaceSettings;
|
||||||
@@ -34,7 +35,7 @@ export const MapRootContent = ({}: MapRootContentProps) => {
|
|||||||
const [showTrackingDialog, setShowTrackingDialog] = useState(false);
|
const [showTrackingDialog, setShowTrackingDialog] = useState(false);
|
||||||
|
|
||||||
/* Important Notice - this solution needs for use one instance of MapInterface */
|
/* Important Notice - this solution needs for use one instance of MapInterface */
|
||||||
const mapInterface = <MapInterface />;
|
const mapInterface = isReady ? <MapInterface /> : null;
|
||||||
|
|
||||||
const handleShowOnTheMap = useCallback(() => setShowOnTheMap(true), []);
|
const handleShowOnTheMap = useCallback(() => setShowOnTheMap(true), []);
|
||||||
const handleShowMapSettings = useCallback(() => setShowMapSettings(true), []);
|
const handleShowMapSettings = useCallback(() => setShowMapSettings(true), []);
|
||||||
@@ -90,6 +91,8 @@ export const MapRootContent = ({}: MapRootContentProps) => {
|
|||||||
{showTrackingDialog && (
|
{showTrackingDialog && (
|
||||||
<TrackingDialog visible={showTrackingDialog} onHide={() => setShowTrackingDialog(false)} />
|
<TrackingDialog visible={showTrackingDialog} onHide={() => setShowTrackingDialog(false)} />
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{hasOldSettings && <OldSettingsDialog />}
|
||||||
</Layout>
|
</Layout>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { Dialog } from 'primereact/dialog';
|
|||||||
import { useCallback, useRef, useState } from 'react';
|
import { useCallback, useRef, useState } from 'react';
|
||||||
import { TabPanel, TabView } from 'primereact/tabview';
|
import { TabPanel, TabView } from 'primereact/tabview';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { OutCommand } from '@/hooks/Mapper/types';
|
import { OutCommand, UserPermission } from '@/hooks/Mapper/types';
|
||||||
import { CONNECTIONS_CHECKBOXES_PROPS, SIGNATURES_CHECKBOXES_PROPS, SYSTEMS_CHECKBOXES_PROPS } from './constants.ts';
|
import { CONNECTIONS_CHECKBOXES_PROPS, SIGNATURES_CHECKBOXES_PROPS, SYSTEMS_CHECKBOXES_PROPS } from './constants.ts';
|
||||||
import {
|
import {
|
||||||
MapSettingsProvider,
|
MapSettingsProvider,
|
||||||
@@ -12,6 +12,10 @@ import {
|
|||||||
import { WidgetsSettings } from './components/WidgetsSettings';
|
import { WidgetsSettings } from './components/WidgetsSettings';
|
||||||
import { CommonSettings } from './components/CommonSettings';
|
import { CommonSettings } from './components/CommonSettings';
|
||||||
import { SettingsListItem } from './types.ts';
|
import { SettingsListItem } from './types.ts';
|
||||||
|
import { ImportExport } from './components/ImportExport.tsx';
|
||||||
|
import { ServerSettings } from './components/ServerSettings.tsx';
|
||||||
|
import { AdminSettings } from './components/AdminSettings.tsx';
|
||||||
|
import { useMapCheckPermissions } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
|
|
||||||
export interface MapSettingsProps {
|
export interface MapSettingsProps {
|
||||||
visible: boolean;
|
visible: boolean;
|
||||||
@@ -23,6 +27,7 @@ export const MapSettingsComp = ({ visible, onHide }: MapSettingsProps) => {
|
|||||||
const { outCommand } = useMapRootState();
|
const { outCommand } = useMapRootState();
|
||||||
|
|
||||||
const { renderSettingItem, setUserRemoteSettings } = useMapSettings();
|
const { renderSettingItem, setUserRemoteSettings } = useMapSettings();
|
||||||
|
const isAdmin = useMapCheckPermissions([UserPermission.ADMIN_MAP]);
|
||||||
|
|
||||||
const refVars = useRef({ outCommand, onHide, visible });
|
const refVars = useRef({ outCommand, onHide, visible });
|
||||||
refVars.current = { outCommand, onHide, visible };
|
refVars.current = { outCommand, onHide, visible };
|
||||||
@@ -57,7 +62,7 @@ export const MapSettingsComp = ({ visible, onHide }: MapSettingsProps) => {
|
|||||||
header="Map user settings"
|
header="Map user settings"
|
||||||
visible
|
visible
|
||||||
draggable={false}
|
draggable={false}
|
||||||
style={{ width: '550px' }}
|
style={{ width: '600px' }}
|
||||||
onShow={handleShow}
|
onShow={handleShow}
|
||||||
onHide={handleHide}
|
onHide={handleHide}
|
||||||
>
|
>
|
||||||
@@ -87,6 +92,20 @@ export const MapSettingsComp = ({ visible, onHide }: MapSettingsProps) => {
|
|||||||
<TabPanel header="Widgets" className="h-full" headerClassName={styles.verticalTabHeader}>
|
<TabPanel header="Widgets" className="h-full" headerClassName={styles.verticalTabHeader}>
|
||||||
<WidgetsSettings />
|
<WidgetsSettings />
|
||||||
</TabPanel>
|
</TabPanel>
|
||||||
|
|
||||||
|
<TabPanel header="Import/Export" className="h-full" headerClassName={styles.verticalTabHeader}>
|
||||||
|
<ImportExport />
|
||||||
|
</TabPanel>
|
||||||
|
|
||||||
|
<TabPanel header="Server Settings" className="h-full" headerClassName="color-warn">
|
||||||
|
<ServerSettings />
|
||||||
|
</TabPanel>
|
||||||
|
|
||||||
|
{isAdmin && (
|
||||||
|
<TabPanel header="Admin Settings" className="h-full" headerClassName="color-warn">
|
||||||
|
<AdminSettings />
|
||||||
|
</TabPanel>
|
||||||
|
)}
|
||||||
</TabView>
|
</TabView>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ import { OutCommand } from '@/hooks/Mapper/types';
|
|||||||
import { PrettySwitchbox } from '@/hooks/Mapper/components/mapRootContent/components/MapSettings/components';
|
import { PrettySwitchbox } from '@/hooks/Mapper/components/mapRootContent/components/MapSettings/components';
|
||||||
import { Dropdown } from 'primereact/dropdown';
|
import { Dropdown } from 'primereact/dropdown';
|
||||||
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { WithChildren } from '@/hooks/Mapper/types/common.ts';
|
||||||
|
|
||||||
type MapSettingsContextType = {
|
type MapSettingsContextType = {
|
||||||
renderSettingItem: (item: SettingsListItem) => ReactNode;
|
renderSettingItem: (item: SettingsListItem) => ReactNode;
|
||||||
@@ -30,7 +31,7 @@ type MapSettingsContextType = {
|
|||||||
|
|
||||||
const MapSettingsContext = createContext<MapSettingsContextType | undefined>(undefined);
|
const MapSettingsContext = createContext<MapSettingsContextType | undefined>(undefined);
|
||||||
|
|
||||||
export const MapSettingsProvider = ({ children }: { children: ReactNode }) => {
|
export const MapSettingsProvider = ({ children }: WithChildren) => {
|
||||||
const {
|
const {
|
||||||
outCommand,
|
outCommand,
|
||||||
storedSettings: { interfaceSettings, setInterfaceSettings },
|
storedSettings: { interfaceSettings, setInterfaceSettings },
|
||||||
|
|||||||
@@ -0,0 +1,128 @@
|
|||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
||||||
|
import { Toast } from 'primereact/toast';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { callToastError, callToastSuccess, callToastWarn } from '@/hooks/Mapper/helpers';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
import { ConfirmPopup } from 'primereact/confirmpopup';
|
||||||
|
import { useConfirmPopup } from '@/hooks/Mapper/hooks';
|
||||||
|
import { MapUserSettings, RemoteAdminSettingsResponse } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import { parseMapUserSettings } from '@/hooks/Mapper/components/helpers';
|
||||||
|
import fastDeepEqual from 'fast-deep-equal';
|
||||||
|
import { useDetectSettingsChanged } from '@/hooks/Mapper/components/hooks';
|
||||||
|
|
||||||
|
export const AdminSettings = () => {
|
||||||
|
const {
|
||||||
|
storedSettings: { getSettingsForExport },
|
||||||
|
outCommand,
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const settingsChanged = useDetectSettingsChanged();
|
||||||
|
|
||||||
|
const [currentRemoteSettings, setCurrentRemoteSettings] = useState<MapUserSettings | null>(null);
|
||||||
|
|
||||||
|
const { cfShow, cfHide, cfVisible, cfRef } = useConfirmPopup();
|
||||||
|
const toast = useRef<Toast | null>(null);
|
||||||
|
|
||||||
|
const hasSettingsForExport = useMemo(() => !!getSettingsForExport(), [getSettingsForExport]);
|
||||||
|
|
||||||
|
const refVars = useRef({ currentRemoteSettings, getSettingsForExport });
|
||||||
|
refVars.current = { currentRemoteSettings, getSettingsForExport };
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const load = async () => {
|
||||||
|
let res: RemoteAdminSettingsResponse | undefined;
|
||||||
|
try {
|
||||||
|
res = await outCommand({ type: OutCommand.getDefaultSettings, data: null });
|
||||||
|
} catch (error) {
|
||||||
|
// do nothing
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!res || res.default_settings == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setCurrentRemoteSettings(parseMapUserSettings(res.default_settings));
|
||||||
|
};
|
||||||
|
|
||||||
|
load();
|
||||||
|
}, [outCommand]);
|
||||||
|
|
||||||
|
const isDirty = useMemo(() => {
|
||||||
|
const { currentRemoteSettings, getSettingsForExport } = refVars.current;
|
||||||
|
const localCurrent = parseMapUserSettings(getSettingsForExport());
|
||||||
|
|
||||||
|
return !fastDeepEqual(currentRemoteSettings, localCurrent);
|
||||||
|
// eslint-disable-next-line
|
||||||
|
}, [settingsChanged, currentRemoteSettings]);
|
||||||
|
|
||||||
|
const handleSync = useCallback(async () => {
|
||||||
|
const settings = getSettingsForExport();
|
||||||
|
|
||||||
|
if (!settings) {
|
||||||
|
callToastWarn(toast.current, 'No settings to save');
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let response: { success: boolean } | undefined;
|
||||||
|
|
||||||
|
try {
|
||||||
|
response = await outCommand({
|
||||||
|
type: OutCommand.saveDefaultSettings,
|
||||||
|
data: { settings },
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
callToastError(toast.current, 'Something went wrong while saving settings');
|
||||||
|
console.error('ERROR: ', err);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!response || !response.success) {
|
||||||
|
callToastError(toast.current, 'Settings not saved - dont not why it');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setCurrentRemoteSettings(parseMapUserSettings(settings));
|
||||||
|
|
||||||
|
callToastSuccess(toast.current, 'Settings saved successfully');
|
||||||
|
}, [getSettingsForExport, outCommand]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="w-full h-full flex flex-col gap-5">
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<div>
|
||||||
|
<Button
|
||||||
|
// @ts-ignore
|
||||||
|
ref={cfRef}
|
||||||
|
onClick={cfShow}
|
||||||
|
icon="pi pi-save"
|
||||||
|
size="small"
|
||||||
|
severity="danger"
|
||||||
|
label="Save as Map Default"
|
||||||
|
className="py-[4px]"
|
||||||
|
disabled={!hasSettingsForExport || !isDirty}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{!isDirty && <span className="text-red-500/70 text-[12px]">*Local and remote are identical.</span>}
|
||||||
|
|
||||||
|
<span className="text-stone-500 text-[12px]">
|
||||||
|
*Will save your current settings as the default for all new users of this map. This action will overwrite any
|
||||||
|
existing default settings.
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Toast ref={toast} />
|
||||||
|
|
||||||
|
<ConfirmPopup
|
||||||
|
target={cfRef.current}
|
||||||
|
visible={cfVisible}
|
||||||
|
onHide={cfHide}
|
||||||
|
message="Your settings will overwrite default. Sure?."
|
||||||
|
icon="pi pi-exclamation-triangle"
|
||||||
|
accept={handleSync}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -7,9 +7,14 @@ import {
|
|||||||
import { useMapSettings } from '@/hooks/Mapper/components/mapRootContent/components/MapSettings/MapSettingsProvider.tsx';
|
import { useMapSettings } from '@/hooks/Mapper/components/mapRootContent/components/MapSettings/MapSettingsProvider.tsx';
|
||||||
import { SettingsListItem } from '@/hooks/Mapper/components/mapRootContent/components/MapSettings/types.ts';
|
import { SettingsListItem } from '@/hooks/Mapper/components/mapRootContent/components/MapSettings/types.ts';
|
||||||
import { useCallback } from 'react';
|
import { useCallback } from 'react';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import { ConfirmPopup } from 'primereact/confirmpopup';
|
||||||
|
import { useConfirmPopup } from '@/hooks/Mapper/hooks';
|
||||||
|
|
||||||
export const CommonSettings = () => {
|
export const CommonSettings = () => {
|
||||||
const { renderSettingItem } = useMapSettings();
|
const { renderSettingItem } = useMapSettings();
|
||||||
|
const { cfShow, cfHide, cfVisible, cfRef } = useConfirmPopup();
|
||||||
|
|
||||||
const renderSettingsList = useCallback(
|
const renderSettingsList = useCallback(
|
||||||
(list: SettingsListItem[]) => {
|
(list: SettingsListItem[]) => {
|
||||||
@@ -18,6 +23,8 @@ export const CommonSettings = () => {
|
|||||||
[renderSettingItem],
|
[renderSettingItem],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const handleResetSettings = () => {};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col h-full gap-1">
|
<div className="flex flex-col h-full gap-1">
|
||||||
<div>
|
<div>
|
||||||
@@ -29,6 +36,33 @@ export const CommonSettings = () => {
|
|||||||
<div className="grid grid-cols-[1fr_auto]">{renderSettingItem(MINI_MAP_PLACEMENT)}</div>
|
<div className="grid grid-cols-[1fr_auto]">{renderSettingItem(MINI_MAP_PLACEMENT)}</div>
|
||||||
<div className="grid grid-cols-[1fr_auto]">{renderSettingItem(PINGS_PLACEMENT)}</div>
|
<div className="grid grid-cols-[1fr_auto]">{renderSettingItem(PINGS_PLACEMENT)}</div>
|
||||||
<div className="grid grid-cols-[1fr_auto]">{renderSettingItem(THEME_SETTING)}</div>
|
<div className="grid grid-cols-[1fr_auto]">{renderSettingItem(THEME_SETTING)}</div>
|
||||||
|
|
||||||
|
<div className="border-b-2 border-dotted border-stone-700/50 h-px my-3" />
|
||||||
|
|
||||||
|
<div className="grid grid-cols-[1fr_auto]">
|
||||||
|
<div />
|
||||||
|
<WdTooltipWrapper content="This dangerous action. And can not be undone" position={TooltipPosition.top}>
|
||||||
|
<Button
|
||||||
|
// @ts-ignore
|
||||||
|
ref={cfRef}
|
||||||
|
className="py-[4px]"
|
||||||
|
onClick={cfShow}
|
||||||
|
outlined
|
||||||
|
size="small"
|
||||||
|
severity="danger"
|
||||||
|
label="Reset Settings"
|
||||||
|
/>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<ConfirmPopup
|
||||||
|
target={cfRef.current}
|
||||||
|
visible={cfVisible}
|
||||||
|
onHide={cfHide}
|
||||||
|
message="All settings for this map will be reset to default."
|
||||||
|
icon="pi pi-exclamation-triangle"
|
||||||
|
accept={handleResetSettings}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -0,0 +1,90 @@
|
|||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useCallback, useRef, useState } from 'react';
|
||||||
|
import { Toast } from 'primereact/toast';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
import { Divider } from 'primereact/divider';
|
||||||
|
import { callToastError, callToastSuccess, callToastWarn } from '@/hooks/Mapper/helpers';
|
||||||
|
|
||||||
|
type SaveDefaultSettingsReturn = { success: boolean; error: string };
|
||||||
|
|
||||||
|
export const DefaultSettings = () => {
|
||||||
|
const {
|
||||||
|
outCommand,
|
||||||
|
storedSettings: { getSettingsForExport },
|
||||||
|
data: { userPermissions },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
const toast = useRef<Toast | null>(null);
|
||||||
|
|
||||||
|
const refVars = useRef({ getSettingsForExport, outCommand });
|
||||||
|
refVars.current = { getSettingsForExport, outCommand };
|
||||||
|
|
||||||
|
const handleSaveAsDefault = useCallback(async () => {
|
||||||
|
const settings = refVars.current.getSettingsForExport();
|
||||||
|
if (!settings) {
|
||||||
|
callToastWarn(toast.current, 'No settings to save');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setLoading(true);
|
||||||
|
|
||||||
|
let response: SaveDefaultSettingsReturn;
|
||||||
|
try {
|
||||||
|
response = await refVars.current.outCommand({
|
||||||
|
type: OutCommand.saveDefaultSettings,
|
||||||
|
data: { settings },
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Save default settings error:', error);
|
||||||
|
callToastError(toast.current, 'Failed to save default settings');
|
||||||
|
setLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (response.success) {
|
||||||
|
callToastSuccess(toast.current, 'Default settings saved successfully');
|
||||||
|
setLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
callToastError(toast.current, response.error || 'Failed to save default settings');
|
||||||
|
setLoading(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
if (!userPermissions?.admin_map) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Divider />
|
||||||
|
<div className="w-full h-full flex flex-col gap-5">
|
||||||
|
<h3 className="text-lg font-semibold">Default Settings (Admin Only)</h3>
|
||||||
|
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<div>
|
||||||
|
<Button
|
||||||
|
onClick={handleSaveAsDefault}
|
||||||
|
icon="pi pi-save"
|
||||||
|
size="small"
|
||||||
|
severity="danger"
|
||||||
|
label="Save as Map Default"
|
||||||
|
className="py-[4px]"
|
||||||
|
loading={loading}
|
||||||
|
disabled={loading}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<span className="text-stone-500 text-[12px]">
|
||||||
|
*Will save your current settings as the default for all new users of this map. This action will overwrite
|
||||||
|
any existing default settings.
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Toast ref={toast} />
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,202 @@
|
|||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useCallback, useMemo, useRef } from 'react';
|
||||||
|
import { Toast } from 'primereact/toast';
|
||||||
|
import { parseMapUserSettings } from '@/hooks/Mapper/components/helpers';
|
||||||
|
import { saveTextFile } from '@/hooks/Mapper/utils/saveToFile.ts';
|
||||||
|
import { SplitButton } from 'primereact/splitbutton';
|
||||||
|
import { loadTextFile } from '@/hooks/Mapper/utils';
|
||||||
|
|
||||||
|
export const ImportExport = () => {
|
||||||
|
const {
|
||||||
|
storedSettings: { getSettingsForExport, applySettings },
|
||||||
|
data: { map_slug },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const toast = useRef<Toast | null>(null);
|
||||||
|
|
||||||
|
const handleImportFromClipboard = useCallback(async () => {
|
||||||
|
const text = await navigator.clipboard.readText();
|
||||||
|
|
||||||
|
if (text == null || text == '') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const parsed = parseMapUserSettings(text);
|
||||||
|
if (applySettings(parsed)) {
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'success',
|
||||||
|
summary: 'Import',
|
||||||
|
detail: 'Map settings was imported successfully.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
window.dispatchEvent(new Event('resize'));
|
||||||
|
}, 100);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'warn',
|
||||||
|
summary: 'Warning',
|
||||||
|
detail: 'Settings already imported. Or something went wrong.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Import from clipboard Error: `, error);
|
||||||
|
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'error',
|
||||||
|
summary: 'Error',
|
||||||
|
detail: 'Some error occurred on import from Clipboard, check console log.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [applySettings]);
|
||||||
|
|
||||||
|
const handleImportFromFile = useCallback(async () => {
|
||||||
|
try {
|
||||||
|
const text = await loadTextFile();
|
||||||
|
|
||||||
|
const parsed = parseMapUserSettings(text);
|
||||||
|
if (applySettings(parsed)) {
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'success',
|
||||||
|
summary: 'Import',
|
||||||
|
detail: 'Map settings was imported successfully.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'warn',
|
||||||
|
summary: 'Warning',
|
||||||
|
detail: 'Settings already imported. Or something went wrong.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Import from file Error: `, error);
|
||||||
|
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'error',
|
||||||
|
summary: 'Error',
|
||||||
|
detail: 'Some error occurred on import from File, check console log.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [applySettings]);
|
||||||
|
|
||||||
|
const handleExportToClipboard = useCallback(async () => {
|
||||||
|
const settings = getSettingsForExport();
|
||||||
|
if (!settings) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await navigator.clipboard.writeText(settings);
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'success',
|
||||||
|
summary: 'Export',
|
||||||
|
detail: 'Map settings copied into clipboard',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Export to clipboard Error: `, error);
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'error',
|
||||||
|
summary: 'Error',
|
||||||
|
detail: 'Some error occurred on copying to clipboard, check console log.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [getSettingsForExport]);
|
||||||
|
|
||||||
|
const handleExportToFile = useCallback(async () => {
|
||||||
|
const settings = getSettingsForExport();
|
||||||
|
if (!settings) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
saveTextFile(`map_settings_${map_slug}.json`, settings);
|
||||||
|
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'success',
|
||||||
|
summary: 'Export to File',
|
||||||
|
detail: 'Map settings successfully saved to file',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Export to cliboard Error: `, error);
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'error',
|
||||||
|
summary: 'Error',
|
||||||
|
detail: 'Some error occurred on saving to file, check console log.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [getSettingsForExport, map_slug]);
|
||||||
|
|
||||||
|
const importItems = useMemo(
|
||||||
|
() => [
|
||||||
|
{
|
||||||
|
label: 'Import from File',
|
||||||
|
icon: 'pi pi-file-import',
|
||||||
|
command: handleImportFromFile,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
[handleImportFromFile],
|
||||||
|
);
|
||||||
|
|
||||||
|
const exportItems = useMemo(
|
||||||
|
() => [
|
||||||
|
{
|
||||||
|
label: 'Export as File',
|
||||||
|
icon: 'pi pi-file-export',
|
||||||
|
command: handleExportToFile,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
[handleExportToFile],
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="w-full h-full flex flex-col gap-5">
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<div>
|
||||||
|
<SplitButton
|
||||||
|
onClick={handleImportFromClipboard}
|
||||||
|
icon="pi pi-download"
|
||||||
|
size="small"
|
||||||
|
severity="warning"
|
||||||
|
label="Import from Clipboard"
|
||||||
|
className="py-[4px]"
|
||||||
|
model={importItems}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<span className="text-stone-500 text-[12px]">
|
||||||
|
*Will read map settings from clipboard. Be careful it could overwrite current settings.
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<div>
|
||||||
|
<SplitButton
|
||||||
|
onClick={handleExportToClipboard}
|
||||||
|
icon="pi pi-upload"
|
||||||
|
size="small"
|
||||||
|
label="Export to Clipboard"
|
||||||
|
className="py-[4px]"
|
||||||
|
model={exportItems}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<span className="text-stone-500 text-[12px]">*Will save map settings to clipboard.</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Toast ref={toast} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,97 @@
|
|||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { useCallback, useEffect, useRef, useState } from 'react';
|
||||||
|
import { Toast } from 'primereact/toast';
|
||||||
|
import { parseMapUserSettings } from '@/hooks/Mapper/components/helpers';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { OutCommand } from '@/hooks/Mapper/types';
|
||||||
|
import { createDefaultWidgetSettings } from '@/hooks/Mapper/mapRootProvider/helpers/createDefaultWidgetSettings.ts';
|
||||||
|
import { callToastSuccess } from '@/hooks/Mapper/helpers';
|
||||||
|
import { ConfirmPopup } from 'primereact/confirmpopup';
|
||||||
|
import { useConfirmPopup } from '@/hooks/Mapper/hooks';
|
||||||
|
import { RemoteAdminSettingsResponse } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
|
||||||
|
export const ServerSettings = () => {
|
||||||
|
const {
|
||||||
|
storedSettings: { applySettings },
|
||||||
|
outCommand,
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const [hasSettings, setHasSettings] = useState(false);
|
||||||
|
const { cfShow, cfHide, cfVisible, cfRef } = useConfirmPopup();
|
||||||
|
const toast = useRef<Toast | null>(null);
|
||||||
|
|
||||||
|
const handleSync = useCallback(async () => {
|
||||||
|
let res: RemoteAdminSettingsResponse | undefined;
|
||||||
|
try {
|
||||||
|
res = await outCommand({ type: OutCommand.getDefaultSettings, data: null });
|
||||||
|
} catch (error) {
|
||||||
|
// do nothing
|
||||||
|
}
|
||||||
|
|
||||||
|
if (res?.default_settings == null) {
|
||||||
|
applySettings(createDefaultWidgetSettings());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
applySettings(parseMapUserSettings(res.default_settings));
|
||||||
|
callToastSuccess(toast.current, 'Settings synchronized successfully');
|
||||||
|
} catch (error) {
|
||||||
|
applySettings(createDefaultWidgetSettings());
|
||||||
|
}
|
||||||
|
}, [applySettings, outCommand]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const load = async () => {
|
||||||
|
let res: RemoteAdminSettingsResponse | undefined;
|
||||||
|
try {
|
||||||
|
res = await outCommand({ type: OutCommand.getDefaultSettings, data: null });
|
||||||
|
} catch (error) {
|
||||||
|
// do nothing
|
||||||
|
}
|
||||||
|
|
||||||
|
if (res?.default_settings == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setHasSettings(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
load();
|
||||||
|
}, [outCommand]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="w-full h-full flex flex-col gap-5">
|
||||||
|
<div className="flex flex-col gap-1">
|
||||||
|
<div>
|
||||||
|
<Button
|
||||||
|
// @ts-ignore
|
||||||
|
ref={cfRef}
|
||||||
|
onClick={cfShow}
|
||||||
|
icon="pi pi-file-import"
|
||||||
|
size="small"
|
||||||
|
severity="warning"
|
||||||
|
label="Sync with Default Settings"
|
||||||
|
className="py-[4px]"
|
||||||
|
disabled={!hasSettings}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{!hasSettings && (
|
||||||
|
<span className="text-red-500/70 text-[12px]">*Default settings was not set by map administrator.</span>
|
||||||
|
)}
|
||||||
|
<span className="text-stone-500 text-[12px]">*Will apply admin settings which set as Default for map.</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Toast ref={toast} />
|
||||||
|
|
||||||
|
<ConfirmPopup
|
||||||
|
target={cfRef.current}
|
||||||
|
visible={cfVisible}
|
||||||
|
onHide={cfHide}
|
||||||
|
message="You lost your current settings. Sure?."
|
||||||
|
icon="pi pi-exclamation-triangle"
|
||||||
|
accept={handleSync}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -28,6 +28,9 @@ export const WidgetsSettings = ({}: WidgetsSettingsProps) => {
|
|||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div className="border-b-2 border-dotted border-stone-700/50 h-px my-3" />
|
||||||
|
|
||||||
<div className="grid grid-cols-[1fr_auto]">
|
<div className="grid grid-cols-[1fr_auto]">
|
||||||
<div />
|
<div />
|
||||||
<Button className="py-[4px]" onClick={resetWidgets} outlined size="small" label="Reset Widgets"></Button>
|
<Button className="py-[4px]" onClick={resetWidgets} outlined size="small" label="Reset Widgets"></Button>
|
||||||
|
|||||||
@@ -0,0 +1,204 @@
|
|||||||
|
import { Dialog } from 'primereact/dialog';
|
||||||
|
import { Button } from 'primereact/button';
|
||||||
|
import { ConfirmPopup } from 'primereact/confirmpopup';
|
||||||
|
import { useCallback, useRef } from 'react';
|
||||||
|
import { MapUserSettings } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import {
|
||||||
|
DEFAULT_KILLS_WIDGET_SETTINGS,
|
||||||
|
DEFAULT_ON_THE_MAP_SETTINGS,
|
||||||
|
DEFAULT_ROUTES_SETTINGS,
|
||||||
|
DEFAULT_WIDGET_LOCAL_SETTINGS,
|
||||||
|
getDefaultWidgetProps,
|
||||||
|
STORED_INTERFACE_DEFAULT_VALUES,
|
||||||
|
} from '@/hooks/Mapper/mapRootProvider/constants.ts';
|
||||||
|
import { DEFAULT_SIGNATURE_SETTINGS } from '@/hooks/Mapper/constants/signatures.ts';
|
||||||
|
import { Toast } from 'primereact/toast';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
import { saveTextFile } from '@/hooks/Mapper/utils';
|
||||||
|
import { useConfirmPopup } from '@/hooks/Mapper/hooks';
|
||||||
|
|
||||||
|
const createSettings = function <T>(lsSettings: string | null, defaultValues: T) {
|
||||||
|
return {
|
||||||
|
version: -1,
|
||||||
|
settings: lsSettings ? JSON.parse(lsSettings) : defaultValues,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export const OldSettingsDialog = () => {
|
||||||
|
const { cfShow, cfHide, cfVisible, cfRef } = useConfirmPopup();
|
||||||
|
const toast = useRef<Toast | null>(null);
|
||||||
|
|
||||||
|
const {
|
||||||
|
storedSettings: { checkOldSettings },
|
||||||
|
data: { map_slug },
|
||||||
|
} = useMapRootState();
|
||||||
|
|
||||||
|
const handleExport = useCallback(
|
||||||
|
async (asFile?: boolean) => {
|
||||||
|
const interfaceSettings = localStorage.getItem('window:interface:settings');
|
||||||
|
const widgetRoutes = localStorage.getItem('window:interface:routes');
|
||||||
|
const widgetLocal = localStorage.getItem('window:interface:local');
|
||||||
|
const widgetKills = localStorage.getItem('kills:widget:settings');
|
||||||
|
const onTheMapOld = localStorage.getItem('window:onTheMap:settings');
|
||||||
|
const widgetsOld = localStorage.getItem('windows:settings:v2');
|
||||||
|
const signatures = localStorage.getItem('wanderer_system_signature_settings_v6_5');
|
||||||
|
|
||||||
|
const out: MapUserSettings = {
|
||||||
|
killsWidget: createSettings(widgetKills, DEFAULT_KILLS_WIDGET_SETTINGS),
|
||||||
|
localWidget: createSettings(widgetLocal, DEFAULT_WIDGET_LOCAL_SETTINGS),
|
||||||
|
widgets: createSettings(widgetsOld, getDefaultWidgetProps()),
|
||||||
|
routes: createSettings(widgetRoutes, DEFAULT_ROUTES_SETTINGS),
|
||||||
|
onTheMap: createSettings(onTheMapOld, DEFAULT_ON_THE_MAP_SETTINGS),
|
||||||
|
signaturesWidget: createSettings(signatures, DEFAULT_SIGNATURE_SETTINGS),
|
||||||
|
interface: createSettings(interfaceSettings, STORED_INTERFACE_DEFAULT_VALUES),
|
||||||
|
};
|
||||||
|
|
||||||
|
if (asFile) {
|
||||||
|
if (!out) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
saveTextFile(`map_settings_${map_slug}.json`, JSON.stringify(out));
|
||||||
|
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'success',
|
||||||
|
summary: 'Export to File',
|
||||||
|
detail: 'Map settings successfully saved to file',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Export to cliboard Error: `, error);
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'error',
|
||||||
|
summary: 'Error',
|
||||||
|
detail: 'Some error occurred on saving to file, check console log.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await navigator.clipboard.writeText(JSON.stringify(out));
|
||||||
|
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'success',
|
||||||
|
summary: 'Export to clipboard',
|
||||||
|
detail: 'Map settings was export successfully.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Export to clipboard Error: `, error);
|
||||||
|
toast.current?.show({
|
||||||
|
severity: 'error',
|
||||||
|
summary: 'Error',
|
||||||
|
detail: 'Some error occurred on copying to clipboard, check console log.',
|
||||||
|
life: 3000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
[map_slug],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleExportClipboard = useCallback(async () => {
|
||||||
|
await handleExport();
|
||||||
|
}, [handleExport]);
|
||||||
|
|
||||||
|
const handleExportAsFile = useCallback(async () => {
|
||||||
|
await handleExport(true);
|
||||||
|
}, [handleExport]);
|
||||||
|
|
||||||
|
const handleProceed = useCallback(() => {
|
||||||
|
localStorage.removeItem('window:interface:settings');
|
||||||
|
localStorage.removeItem('window:interface:routes');
|
||||||
|
localStorage.removeItem('window:interface:local');
|
||||||
|
localStorage.removeItem('kills:widget:settings');
|
||||||
|
localStorage.removeItem('window:onTheMap:settings');
|
||||||
|
localStorage.removeItem('windows:settings:v2');
|
||||||
|
localStorage.removeItem('wanderer_system_signature_settings_v6_5');
|
||||||
|
|
||||||
|
checkOldSettings();
|
||||||
|
}, [checkOldSettings]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Dialog
|
||||||
|
header={
|
||||||
|
<div className="dialog-header">
|
||||||
|
<span className="pointer-events-none">Old settings detected!</span>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
draggable={false}
|
||||||
|
resizable={false}
|
||||||
|
closable={false}
|
||||||
|
visible
|
||||||
|
onHide={() => null}
|
||||||
|
className="w-[640px] h-[400px] text-text-color min-h-0"
|
||||||
|
footer={
|
||||||
|
<div className="flex items-center justify-end">
|
||||||
|
<Button
|
||||||
|
// @ts-ignore
|
||||||
|
ref={cfRef}
|
||||||
|
onClick={cfShow}
|
||||||
|
icon="pi pi-exclamation-triangle"
|
||||||
|
size="small"
|
||||||
|
severity="warning"
|
||||||
|
label="Proceed"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div className="w-full h-full flex flex-col gap-1 items-center justify-center text-stone-400 text-[15px]">
|
||||||
|
<span>
|
||||||
|
We detected <span className="text-orange-400">deprecated</span> settings saved in your browser.
|
||||||
|
</span>
|
||||||
|
<span>
|
||||||
|
Now we will give you ability to make <span className="text-orange-400">export</span> your old settings.
|
||||||
|
</span>
|
||||||
|
<span>
|
||||||
|
After click: all settings will saved in your <span className="text-orange-400">clipboard</span>.
|
||||||
|
</span>
|
||||||
|
<span>
|
||||||
|
Then you need to go into <span className="text-orange-400">Map Settings</span> and click{' '}
|
||||||
|
<span className="text-orange-400">Import from clipboard</span>
|
||||||
|
</span>
|
||||||
|
<div className="h-[30px]"></div>
|
||||||
|
|
||||||
|
<div className="flex items-center gap-3">
|
||||||
|
<Button
|
||||||
|
onClick={handleExportClipboard}
|
||||||
|
icon="pi pi-copy"
|
||||||
|
size="small"
|
||||||
|
severity="info"
|
||||||
|
label="Export to Clipboard"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
onClick={handleExportAsFile}
|
||||||
|
icon="pi pi-download"
|
||||||
|
size="small"
|
||||||
|
severity="info"
|
||||||
|
label="Export as File"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<span className="text-stone-600 text-[12px]">*You will see this dialog until click Export.</span>
|
||||||
|
</div>
|
||||||
|
</Dialog>
|
||||||
|
|
||||||
|
<ConfirmPopup
|
||||||
|
target={cfRef.current}
|
||||||
|
visible={cfVisible}
|
||||||
|
onHide={cfHide}
|
||||||
|
message="After click dialog will disappear. Ready?"
|
||||||
|
icon="pi pi-exclamation-triangle"
|
||||||
|
accept={handleProceed}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Toast ref={toast} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -7,25 +7,14 @@ import { VirtualScroller, VirtualScrollerTemplateOptions } from 'primereact/virt
|
|||||||
import clsx from 'clsx';
|
import clsx from 'clsx';
|
||||||
import { CharacterTypeRaw, WithIsOwnCharacter } from '@/hooks/Mapper/types';
|
import { CharacterTypeRaw, WithIsOwnCharacter } from '@/hooks/Mapper/types';
|
||||||
import { CharacterCard, TooltipPosition, WdCheckbox, WdImageSize, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
import { CharacterCard, TooltipPosition, WdCheckbox, WdImageSize, WdImgButton } from '@/hooks/Mapper/components/ui-kit';
|
||||||
import useLocalStorageState from 'use-local-storage-state';
|
|
||||||
import { useMapCheckPermissions, useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
import { useMapCheckPermissions, useMapGetOption } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
import { UserPermission } from '@/hooks/Mapper/types/permissions.ts';
|
import { UserPermission } from '@/hooks/Mapper/types/permissions.ts';
|
||||||
import { InputText } from 'primereact/inputtext';
|
import { InputText } from 'primereact/inputtext';
|
||||||
import { IconField } from 'primereact/iconfield';
|
import { IconField } from 'primereact/iconfield';
|
||||||
|
|
||||||
type WindowLocalSettingsType = {
|
|
||||||
compact: boolean;
|
|
||||||
hideOffline: boolean;
|
|
||||||
version: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
const STORED_DEFAULT_VALUES: WindowLocalSettingsType = {
|
|
||||||
compact: true,
|
|
||||||
hideOffline: false,
|
|
||||||
version: 0,
|
|
||||||
};
|
|
||||||
|
|
||||||
const itemTemplate = (item: CharacterTypeRaw & WithIsOwnCharacter, options: VirtualScrollerTemplateOptions) => {
|
const itemTemplate = (item: CharacterTypeRaw & WithIsOwnCharacter, options: VirtualScrollerTemplateOptions) => {
|
||||||
|
const showAllyLogoPlaceholder = options.props.items?.some(x => x.alliance_id != null);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
className={clsx(classes.CharacterRow, 'w-full box-border px-2 py-1', {
|
className={clsx(classes.CharacterRow, 'w-full box-border px-2 py-1', {
|
||||||
@@ -35,7 +24,15 @@ const itemTemplate = (item: CharacterTypeRaw & WithIsOwnCharacter, options: Virt
|
|||||||
})}
|
})}
|
||||||
style={{ height: options.props.itemSize + 'px' }}
|
style={{ height: options.props.itemSize + 'px' }}
|
||||||
>
|
>
|
||||||
<CharacterCard showCorporationLogo showAllyLogo showSystem showTicker showShip {...item} />
|
<CharacterCard
|
||||||
|
showCorporationLogo
|
||||||
|
showAllyLogo
|
||||||
|
showAllyLogoPlaceholder={showAllyLogoPlaceholder}
|
||||||
|
showSystem
|
||||||
|
showTicker
|
||||||
|
showShip
|
||||||
|
{...item}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
@@ -48,14 +45,11 @@ export interface OnTheMapProps {
|
|||||||
export const OnTheMap = ({ show, onHide }: OnTheMapProps) => {
|
export const OnTheMap = ({ show, onHide }: OnTheMapProps) => {
|
||||||
const {
|
const {
|
||||||
data: { characters, userCharacters },
|
data: { characters, userCharacters },
|
||||||
|
storedSettings: { settingsOnTheMap, settingsOnTheMapUpdate },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const [searchVal, setSearchVal] = useState('');
|
const [searchVal, setSearchVal] = useState('');
|
||||||
|
|
||||||
const [settings, setSettings] = useLocalStorageState<WindowLocalSettingsType>('window:onTheMap:settings', {
|
|
||||||
defaultValue: STORED_DEFAULT_VALUES,
|
|
||||||
});
|
|
||||||
|
|
||||||
const restrictOfflineShowing = useMapGetOption('restrict_offline_showing');
|
const restrictOfflineShowing = useMapGetOption('restrict_offline_showing');
|
||||||
const isAdminOrManager = useMapCheckPermissions([UserPermission.MANAGE_MAP]);
|
const isAdminOrManager = useMapCheckPermissions([UserPermission.MANAGE_MAP]);
|
||||||
|
|
||||||
@@ -107,12 +101,12 @@ export const OnTheMap = ({ show, onHide }: OnTheMapProps) => {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (showOffline && !settings.hideOffline) {
|
if (showOffline && !settingsOnTheMap.hideOffline) {
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
return out.filter(x => x.online);
|
return out.filter(x => x.online);
|
||||||
}, [showOffline, searchVal, characters, settings.hideOffline, userCharacters]);
|
}, [showOffline, searchVal, characters, settingsOnTheMap.hideOffline, userCharacters]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Sidebar
|
<Sidebar
|
||||||
@@ -153,9 +147,11 @@ export const OnTheMap = ({ show, onHide }: OnTheMapProps) => {
|
|||||||
size="m"
|
size="m"
|
||||||
labelSide="left"
|
labelSide="left"
|
||||||
label={'Hide offline'}
|
label={'Hide offline'}
|
||||||
value={settings.hideOffline}
|
value={settingsOnTheMap.hideOffline}
|
||||||
classNameLabel="text-stone-400 hover:text-stone-200 transition duration-300"
|
classNameLabel="text-stone-400 hover:text-stone-200 transition duration-300"
|
||||||
onChange={() => setSettings(() => ({ ...settings, hideOffline: !settings.hideOffline }))}
|
onChange={() =>
|
||||||
|
settingsOnTheMapUpdate(() => ({ ...settingsOnTheMap, hideOffline: !settingsOnTheMap.hideOffline }))
|
||||||
|
}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -0,0 +1,49 @@
|
|||||||
|
import { TooltipPosition, WdTooltipWrapper } from '@/hooks/Mapper/components/ui-kit';
|
||||||
|
import useLocalStorageState from 'use-local-storage-state';
|
||||||
|
import { useMapRootState } from '@/hooks/Mapper/mapRootProvider';
|
||||||
|
|
||||||
|
export const DebugComponent = () => {
|
||||||
|
const { outCommand } = useMapRootState();
|
||||||
|
|
||||||
|
const [record, setRecord] = useLocalStorageState<boolean>('record', {
|
||||||
|
defaultValue: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const [recordsList] = useLocalStorageState<{ type; data }[]>('recordsList', {
|
||||||
|
defaultValue: [],
|
||||||
|
});
|
||||||
|
|
||||||
|
const handleRunSavedEvents = () => {
|
||||||
|
recordsList.forEach(record => outCommand(record));
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<WdTooltipWrapper content="Run saved events" position={TooltipPosition.left}>
|
||||||
|
<button
|
||||||
|
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
|
||||||
|
type="button"
|
||||||
|
onClick={handleRunSavedEvents}
|
||||||
|
disabled={recordsList.length === 0 || record}
|
||||||
|
>
|
||||||
|
<i className="pi pi-forward"></i>
|
||||||
|
</button>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
|
||||||
|
<WdTooltipWrapper content="Record" position={TooltipPosition.left}>
|
||||||
|
<button
|
||||||
|
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
|
||||||
|
type="button"
|
||||||
|
onClick={() => setRecord(x => !x)}
|
||||||
|
>
|
||||||
|
{!record ? (
|
||||||
|
<i className="pi pi-play-circle text-green-500"></i>
|
||||||
|
) : (
|
||||||
|
<i className="pi pi-stop-circle text-red-500"></i>
|
||||||
|
)}
|
||||||
|
</button>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -7,6 +7,7 @@ import { TooltipPosition } from '@/hooks/Mapper/components/ui-kit';
|
|||||||
|
|
||||||
import { useMapCheckPermissions } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
import { useMapCheckPermissions } from '@/hooks/Mapper/mapRootProvider/hooks/api';
|
||||||
import { UserPermission } from '@/hooks/Mapper/types/permissions.ts';
|
import { UserPermission } from '@/hooks/Mapper/types/permissions.ts';
|
||||||
|
// import { DebugComponent } from '@/hooks/Mapper/components/mapRootContent/components/RightBar/DebugComponent.tsx';
|
||||||
|
|
||||||
interface RightBarProps {
|
interface RightBarProps {
|
||||||
onShowOnTheMap?: () => void;
|
onShowOnTheMap?: () => void;
|
||||||
@@ -79,6 +80,9 @@ export const RightBar = ({
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-col items-center mb-2 gap-1">
|
<div className="flex flex-col items-center mb-2 gap-1">
|
||||||
|
{/* TODO - do not delete this code needs for debug */}
|
||||||
|
{/*<DebugComponent />*/}
|
||||||
|
|
||||||
<WdTooltipWrapper content="Map user settings" position={TooltipPosition.left}>
|
<WdTooltipWrapper content="Map user settings" position={TooltipPosition.left}>
|
||||||
<button
|
<button
|
||||||
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
|
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ export const MapWrapper = () => {
|
|||||||
linkSignatureToSystem,
|
linkSignatureToSystem,
|
||||||
systemSignatures,
|
systemSignatures,
|
||||||
},
|
},
|
||||||
storedSettings: { interfaceSettings },
|
storedSettings: { interfaceSettings, settingsLocal },
|
||||||
} = useMapRootState();
|
} = useMapRootState();
|
||||||
|
|
||||||
const {
|
const {
|
||||||
@@ -181,17 +181,20 @@ export const MapWrapper = () => {
|
|||||||
ref.current.systemContextProps.systemId && setOpenSettings(ref.current.systemContextProps.systemId);
|
ref.current.systemContextProps.systemId && setOpenSettings(ref.current.systemContextProps.systemId);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const handleTogglePing = useCallback(async (type: PingType, solar_system_id: string, hasPing: boolean) => {
|
const handleTogglePing = useCallback(
|
||||||
|
async (type: PingType, solar_system_id: string, ping_id: string | undefined, hasPing: boolean) => {
|
||||||
if (hasPing) {
|
if (hasPing) {
|
||||||
await outCommand({
|
await outCommand({
|
||||||
type: OutCommand.cancelPing,
|
type: OutCommand.cancelPing,
|
||||||
data: { type, solar_system_id: solar_system_id },
|
data: { type, id: ping_id },
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
setOpenPing({ type, solar_system_id });
|
setOpenPing({ type, solar_system_id });
|
||||||
}, []);
|
},
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
|
||||||
const handleCustomLabelDialog = useCallback(() => {
|
const handleCustomLabelDialog = useCallback(() => {
|
||||||
const { systemContextProps } = ref.current;
|
const { systemContextProps } = ref.current;
|
||||||
@@ -254,6 +257,7 @@ export const MapWrapper = () => {
|
|||||||
pings={pings}
|
pings={pings}
|
||||||
onAddSystem={onAddSystem}
|
onAddSystem={onAddSystem}
|
||||||
minimapPlacement={minimapPosition}
|
minimapPlacement={minimapPosition}
|
||||||
|
localShowShipName={settingsLocal.showShipName}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
{openSettings != null && (
|
{openSettings != null && (
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ export type CharacterCardProps = {
|
|||||||
useSystemsCache?: boolean;
|
useSystemsCache?: boolean;
|
||||||
showCorporationLogo?: boolean;
|
showCorporationLogo?: boolean;
|
||||||
showAllyLogo?: boolean;
|
showAllyLogo?: boolean;
|
||||||
|
showAllyLogoPlaceholder?: boolean;
|
||||||
simpleMode?: boolean;
|
simpleMode?: boolean;
|
||||||
} & WithIsOwnCharacter &
|
} & WithIsOwnCharacter &
|
||||||
WithClassName;
|
WithClassName;
|
||||||
@@ -47,6 +48,7 @@ export const CharacterCard = ({
|
|||||||
showShipName,
|
showShipName,
|
||||||
showCorporationLogo,
|
showCorporationLogo,
|
||||||
showAllyLogo,
|
showAllyLogo,
|
||||||
|
showAllyLogoPlaceholder,
|
||||||
showTicker,
|
showTicker,
|
||||||
useSystemsCache,
|
useSystemsCache,
|
||||||
className,
|
className,
|
||||||
@@ -217,6 +219,18 @@ export const CharacterCard = ({
|
|||||||
/>
|
/>
|
||||||
</WdTooltipWrapper>
|
</WdTooltipWrapper>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
{showAllyLogo && showAllyLogoPlaceholder && !char.alliance_id && (
|
||||||
|
<WdTooltipWrapper position={TooltipPosition.top} content="No alliance">
|
||||||
|
<span
|
||||||
|
className={clsx(
|
||||||
|
'min-w-[33px] min-h-[33px] w-[33px] h-[33px]',
|
||||||
|
'flex transition-[border-color,opacity] duration-250 rounded-none',
|
||||||
|
'wd-bg-default',
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</WdTooltipWrapper>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-col flex-grow overflow-hidden w-[50px]">
|
<div className="flex flex-col flex-grow overflow-hidden w-[50px]">
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ export enum Regions {
|
|||||||
Solitude = 10000044,
|
Solitude = 10000044,
|
||||||
TashMurkon = 10000020,
|
TashMurkon = 10000020,
|
||||||
VergeVendor = 10000068,
|
VergeVendor = 10000068,
|
||||||
|
Pochven = 10000070,
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum Spaces {
|
export enum Spaces {
|
||||||
@@ -40,6 +41,7 @@ export enum Spaces {
|
|||||||
'Gallente' = 'Gallente',
|
'Gallente' = 'Gallente',
|
||||||
'Matar' = 'Matar',
|
'Matar' = 'Matar',
|
||||||
'Amarr' = 'Amarr',
|
'Amarr' = 'Amarr',
|
||||||
|
'Pochven' = 'Pochven',
|
||||||
}
|
}
|
||||||
|
|
||||||
export const REGIONS_MAP: Record<number, Spaces> = {
|
export const REGIONS_MAP: Record<number, Spaces> = {
|
||||||
@@ -66,6 +68,7 @@ export const REGIONS_MAP: Record<number, Spaces> = {
|
|||||||
[Regions.Solitude]: Spaces.Gallente,
|
[Regions.Solitude]: Spaces.Gallente,
|
||||||
[Regions.TashMurkon]: Spaces.Amarr,
|
[Regions.TashMurkon]: Spaces.Amarr,
|
||||||
[Regions.VergeVendor]: Spaces.Gallente,
|
[Regions.VergeVendor]: Spaces.Gallente,
|
||||||
|
[Regions.Pochven]: Spaces.Pochven,
|
||||||
};
|
};
|
||||||
|
|
||||||
export type K162Type = {
|
export type K162Type = {
|
||||||
|
|||||||
71
assets/js/hooks/Mapper/constants/signatures.ts
Normal file
71
assets/js/hooks/Mapper/constants/signatures.ts
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
import { SignatureGroup, SignatureKind } from '@/hooks/Mapper/types';
|
||||||
|
|
||||||
|
export const SIGNATURE_WINDOW_ID = 'system_signatures_window';
|
||||||
|
|
||||||
|
export enum SIGNATURES_DELETION_TIMING {
|
||||||
|
IMMEDIATE,
|
||||||
|
DEFAULT,
|
||||||
|
EXTENDED,
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum SETTINGS_KEYS {
|
||||||
|
SORT_FIELD = 'sortField',
|
||||||
|
SORT_ORDER = 'sortOrder',
|
||||||
|
|
||||||
|
SHOW_DESCRIPTION_COLUMN = 'show_description_column',
|
||||||
|
SHOW_UPDATED_COLUMN = 'show_updated_column',
|
||||||
|
SHOW_CHARACTER_COLUMN = 'show_character_column',
|
||||||
|
LAZY_DELETE_SIGNATURES = 'lazy_delete_signatures',
|
||||||
|
KEEP_LAZY_DELETE = 'keep_lazy_delete_enabled',
|
||||||
|
DELETION_TIMING = 'deletion_timing',
|
||||||
|
COLOR_BY_TYPE = 'color_by_type',
|
||||||
|
SHOW_CHARACTER_PORTRAIT = 'show_character_portrait',
|
||||||
|
|
||||||
|
// From SignatureKind
|
||||||
|
COSMIC_ANOMALY = SignatureKind.CosmicAnomaly,
|
||||||
|
COSMIC_SIGNATURE = SignatureKind.CosmicSignature,
|
||||||
|
DEPLOYABLE = SignatureKind.Deployable,
|
||||||
|
STRUCTURE = SignatureKind.Structure,
|
||||||
|
STARBASE = SignatureKind.Starbase,
|
||||||
|
SHIP = SignatureKind.Ship,
|
||||||
|
DRONE = SignatureKind.Drone,
|
||||||
|
|
||||||
|
// From SignatureGroup
|
||||||
|
WORMHOLE = SignatureGroup.Wormhole,
|
||||||
|
RELIC_SITE = SignatureGroup.RelicSite,
|
||||||
|
DATA_SITE = SignatureGroup.DataSite,
|
||||||
|
ORE_SITE = SignatureGroup.OreSite,
|
||||||
|
GAS_SITE = SignatureGroup.GasSite,
|
||||||
|
COMBAT_SITE = SignatureGroup.CombatSite,
|
||||||
|
}
|
||||||
|
|
||||||
|
export type SignatureSettingsType = { [key in SETTINGS_KEYS]?: unknown };
|
||||||
|
|
||||||
|
export const DEFAULT_SIGNATURE_SETTINGS: SignatureSettingsType = {
|
||||||
|
[SETTINGS_KEYS.SORT_FIELD]: 'inserted_at',
|
||||||
|
[SETTINGS_KEYS.SORT_ORDER]: -1,
|
||||||
|
|
||||||
|
[SETTINGS_KEYS.SHOW_UPDATED_COLUMN]: true,
|
||||||
|
[SETTINGS_KEYS.SHOW_DESCRIPTION_COLUMN]: true,
|
||||||
|
[SETTINGS_KEYS.SHOW_CHARACTER_COLUMN]: true,
|
||||||
|
[SETTINGS_KEYS.LAZY_DELETE_SIGNATURES]: true,
|
||||||
|
[SETTINGS_KEYS.KEEP_LAZY_DELETE]: false,
|
||||||
|
[SETTINGS_KEYS.DELETION_TIMING]: SIGNATURES_DELETION_TIMING.DEFAULT,
|
||||||
|
[SETTINGS_KEYS.COLOR_BY_TYPE]: true,
|
||||||
|
[SETTINGS_KEYS.SHOW_CHARACTER_PORTRAIT]: true,
|
||||||
|
|
||||||
|
[SETTINGS_KEYS.COSMIC_ANOMALY]: true,
|
||||||
|
[SETTINGS_KEYS.COSMIC_SIGNATURE]: true,
|
||||||
|
[SETTINGS_KEYS.DEPLOYABLE]: true,
|
||||||
|
[SETTINGS_KEYS.STRUCTURE]: true,
|
||||||
|
[SETTINGS_KEYS.STARBASE]: true,
|
||||||
|
[SETTINGS_KEYS.SHIP]: true,
|
||||||
|
[SETTINGS_KEYS.DRONE]: true,
|
||||||
|
|
||||||
|
[SETTINGS_KEYS.WORMHOLE]: true,
|
||||||
|
[SETTINGS_KEYS.RELIC_SITE]: true,
|
||||||
|
[SETTINGS_KEYS.DATA_SITE]: true,
|
||||||
|
[SETTINGS_KEYS.ORE_SITE]: true,
|
||||||
|
[SETTINGS_KEYS.GAS_SITE]: true,
|
||||||
|
[SETTINGS_KEYS.COMBAT_SITE]: true,
|
||||||
|
};
|
||||||
11
assets/js/hooks/Mapper/helpers/getFormattedTime.ts
Normal file
11
assets/js/hooks/Mapper/helpers/getFormattedTime.ts
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
export function getFormattedTime() {
|
||||||
|
const now = new Date();
|
||||||
|
|
||||||
|
const hours = String(now.getHours()).padStart(2, '0');
|
||||||
|
const minutes = String(now.getMinutes()).padStart(2, '0');
|
||||||
|
const seconds = String(now.getSeconds()).padStart(2, '0');
|
||||||
|
|
||||||
|
const ms = String(now.getMilliseconds() + 1000).slice(1);
|
||||||
|
|
||||||
|
return `${hours}:${minutes}:${seconds} ${ms}`;
|
||||||
|
}
|
||||||
@@ -2,3 +2,4 @@ export * from './sortWHClasses';
|
|||||||
export * from './parseSignatures';
|
export * from './parseSignatures';
|
||||||
export * from './getSystemById';
|
export * from './getSystemById';
|
||||||
export * from './getEveImageUrl';
|
export * from './getEveImageUrl';
|
||||||
|
export * from './toastHelpers';
|
||||||
|
|||||||
28
assets/js/hooks/Mapper/helpers/toastHelpers.ts
Normal file
28
assets/js/hooks/Mapper/helpers/toastHelpers.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
import { Toast } from 'primereact/toast';
|
||||||
|
|
||||||
|
export const callToastWarn = (toast: Toast | null, msg: string, life = 3000) => {
|
||||||
|
toast?.show({
|
||||||
|
severity: 'warn',
|
||||||
|
summary: 'Warning',
|
||||||
|
detail: msg,
|
||||||
|
life,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const callToastError = (toast: Toast | null, msg: string, life = 3000) => {
|
||||||
|
toast?.show({
|
||||||
|
severity: 'error',
|
||||||
|
summary: 'Error',
|
||||||
|
detail: msg,
|
||||||
|
life,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const callToastSuccess = (toast: Toast | null, msg: string, life = 3000) => {
|
||||||
|
toast?.show({
|
||||||
|
severity: 'success',
|
||||||
|
summary: 'Success',
|
||||||
|
detail: msg,
|
||||||
|
life,
|
||||||
|
});
|
||||||
|
};
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
export * from './useActualizeSettings';
|
|
||||||
export * from './useClipboard';
|
export * from './useClipboard';
|
||||||
export * from './useHotkey';
|
export * from './useHotkey';
|
||||||
export * from './usePageVisibility';
|
export * from './usePageVisibility';
|
||||||
export * from './useSkipContextMenu';
|
export * from './useSkipContextMenu';
|
||||||
export * from './useThrottle';
|
export * from './useThrottle';
|
||||||
|
export * from './useConfirmPopup';
|
||||||
|
|||||||
@@ -1,23 +0,0 @@
|
|||||||
import { useEffect } from 'react';
|
|
||||||
|
|
||||||
type Settings = Record<string, unknown>;
|
|
||||||
export const useActualizeSettings = <T extends Settings>(defaultVals: T, vals: T, setVals: (newVals: T) => void) => {
|
|
||||||
useEffect(() => {
|
|
||||||
let foundNew = false;
|
|
||||||
const newVals = Object.keys(defaultVals).reduce((acc, x) => {
|
|
||||||
if (Object.keys(acc).includes(x)) {
|
|
||||||
return acc;
|
|
||||||
}
|
|
||||||
|
|
||||||
foundNew = true;
|
|
||||||
|
|
||||||
// @ts-ignore
|
|
||||||
return { ...acc, [x]: defaultVals[x] };
|
|
||||||
}, vals);
|
|
||||||
|
|
||||||
if (foundNew) {
|
|
||||||
setVals(newVals);
|
|
||||||
}
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
}, []);
|
|
||||||
};
|
|
||||||
10
assets/js/hooks/Mapper/hooks/useConfirmPopup.ts
Normal file
10
assets/js/hooks/Mapper/hooks/useConfirmPopup.ts
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
import { useCallback, useRef, useState } from 'react';
|
||||||
|
|
||||||
|
export const useConfirmPopup = () => {
|
||||||
|
const cfRef = useRef<HTMLElement>();
|
||||||
|
const [cfVisible, setCfVisible] = useState(false);
|
||||||
|
const cfShow = useCallback(() => setCfVisible(true), []);
|
||||||
|
const cfHide = useCallback(() => setCfVisible(false), []);
|
||||||
|
|
||||||
|
return { cfRef, cfVisible, cfShow, cfHide };
|
||||||
|
};
|
||||||
@@ -1,11 +1,12 @@
|
|||||||
import { useState, useEffect } from 'react';
|
import { useState, useEffect } from 'react';
|
||||||
|
|
||||||
function usePageVisibility() {
|
function usePageVisibility() {
|
||||||
const [isVisible, setIsVisible] = useState(!document.hidden);
|
const getIsVisible = () => !document.hidden;
|
||||||
|
const [isVisible, setIsVisible] = useState(getIsVisible());
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const handleVisibilityChange = () => {
|
const handleVisibilityChange = () => {
|
||||||
setIsVisible(!document.hidden);
|
setIsVisible(getIsVisible());
|
||||||
};
|
};
|
||||||
|
|
||||||
document.addEventListener('visibilitychange', handleVisibilityChange);
|
document.addEventListener('visibilitychange', handleVisibilityChange);
|
||||||
|
|||||||
@@ -19,10 +19,24 @@ import {
|
|||||||
} from '@/hooks/Mapper/mapRootProvider/hooks/useStoreWidgets.ts';
|
} from '@/hooks/Mapper/mapRootProvider/hooks/useStoreWidgets.ts';
|
||||||
import { WindowsManagerOnChange } from '@/hooks/Mapper/components/ui-kit/WindowManager';
|
import { WindowsManagerOnChange } from '@/hooks/Mapper/components/ui-kit/WindowManager';
|
||||||
import { DetailedKill } from '../types/kills';
|
import { DetailedKill } from '../types/kills';
|
||||||
import { InterfaceStoredSettings, RoutesType } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
import {
|
||||||
import { DEFAULT_ROUTES_SETTINGS, STORED_INTERFACE_DEFAULT_VALUES } from '@/hooks/Mapper/mapRootProvider/constants.ts';
|
InterfaceStoredSettings,
|
||||||
|
KillsWidgetSettings,
|
||||||
|
LocalWidgetSettings,
|
||||||
|
MapUserSettings,
|
||||||
|
OnTheMapSettingsType,
|
||||||
|
RoutesType,
|
||||||
|
} from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import {
|
||||||
|
DEFAULT_KILLS_WIDGET_SETTINGS,
|
||||||
|
DEFAULT_ON_THE_MAP_SETTINGS,
|
||||||
|
DEFAULT_ROUTES_SETTINGS,
|
||||||
|
DEFAULT_WIDGET_LOCAL_SETTINGS,
|
||||||
|
STORED_INTERFACE_DEFAULT_VALUES,
|
||||||
|
} from '@/hooks/Mapper/mapRootProvider/constants.ts';
|
||||||
import { useMapUserSettings } from '@/hooks/Mapper/mapRootProvider/hooks/useMapUserSettings.ts';
|
import { useMapUserSettings } from '@/hooks/Mapper/mapRootProvider/hooks/useMapUserSettings.ts';
|
||||||
import { useGlobalHooks } from '@/hooks/Mapper/mapRootProvider/hooks/useGlobalHooks.ts';
|
import { useGlobalHooks } from '@/hooks/Mapper/mapRootProvider/hooks/useGlobalHooks.ts';
|
||||||
|
import { DEFAULT_SIGNATURE_SETTINGS, SignatureSettingsType } from '@/hooks/Mapper/constants/signatures';
|
||||||
|
|
||||||
export type MapRootData = MapUnionTypes & {
|
export type MapRootData = MapUnionTypes & {
|
||||||
selectedSystems: string[];
|
selectedSystems: string[];
|
||||||
@@ -36,6 +50,7 @@ export type MapRootData = MapUnionTypes & {
|
|||||||
};
|
};
|
||||||
trackingCharactersData: TrackingCharacter[];
|
trackingCharactersData: TrackingCharacter[];
|
||||||
loadingPublicRoutes: boolean;
|
loadingPublicRoutes: boolean;
|
||||||
|
map_slug: string | null;
|
||||||
};
|
};
|
||||||
|
|
||||||
const INITIAL_DATA: MapRootData = {
|
const INITIAL_DATA: MapRootData = {
|
||||||
@@ -70,6 +85,7 @@ const INITIAL_DATA: MapRootData = {
|
|||||||
followingCharacterEveId: null,
|
followingCharacterEveId: null,
|
||||||
pings: [],
|
pings: [],
|
||||||
loadingPublicRoutes: false,
|
loadingPublicRoutes: false,
|
||||||
|
map_slug: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
export enum InterfaceStoredSettingsProps {
|
export enum InterfaceStoredSettingsProps {
|
||||||
@@ -103,6 +119,20 @@ export interface MapRootContextProps {
|
|||||||
setInterfaceSettings: Dispatch<SetStateAction<InterfaceStoredSettings>>;
|
setInterfaceSettings: Dispatch<SetStateAction<InterfaceStoredSettings>>;
|
||||||
settingsRoutes: RoutesType;
|
settingsRoutes: RoutesType;
|
||||||
settingsRoutesUpdate: Dispatch<SetStateAction<RoutesType>>;
|
settingsRoutesUpdate: Dispatch<SetStateAction<RoutesType>>;
|
||||||
|
settingsLocal: LocalWidgetSettings;
|
||||||
|
settingsLocalUpdate: Dispatch<SetStateAction<LocalWidgetSettings>>;
|
||||||
|
settingsSignatures: SignatureSettingsType;
|
||||||
|
settingsSignaturesUpdate: Dispatch<SetStateAction<SignatureSettingsType>>;
|
||||||
|
settingsOnTheMap: OnTheMapSettingsType;
|
||||||
|
settingsOnTheMapUpdate: Dispatch<SetStateAction<OnTheMapSettingsType>>;
|
||||||
|
settingsKills: KillsWidgetSettings;
|
||||||
|
settingsKillsUpdate: Dispatch<SetStateAction<KillsWidgetSettings>>;
|
||||||
|
isReady: boolean;
|
||||||
|
hasOldSettings: boolean;
|
||||||
|
getSettingsForExport(): string | undefined;
|
||||||
|
applySettings(settings: MapUserSettings): boolean;
|
||||||
|
resetSettings(settings: MapUserSettings): void;
|
||||||
|
checkOldSettings(): void;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -134,6 +164,20 @@ const MapRootContext = createContext<MapRootContextProps>({
|
|||||||
setInterfaceSettings: () => null,
|
setInterfaceSettings: () => null,
|
||||||
settingsRoutes: DEFAULT_ROUTES_SETTINGS,
|
settingsRoutes: DEFAULT_ROUTES_SETTINGS,
|
||||||
settingsRoutesUpdate: () => null,
|
settingsRoutesUpdate: () => null,
|
||||||
|
settingsLocal: DEFAULT_WIDGET_LOCAL_SETTINGS,
|
||||||
|
settingsLocalUpdate: () => null,
|
||||||
|
settingsSignatures: DEFAULT_SIGNATURE_SETTINGS,
|
||||||
|
settingsSignaturesUpdate: () => null,
|
||||||
|
settingsOnTheMap: DEFAULT_ON_THE_MAP_SETTINGS,
|
||||||
|
settingsOnTheMapUpdate: () => null,
|
||||||
|
settingsKills: DEFAULT_KILLS_WIDGET_SETTINGS,
|
||||||
|
settingsKillsUpdate: () => null,
|
||||||
|
isReady: false,
|
||||||
|
hasOldSettings: false,
|
||||||
|
getSettingsForExport: () => '',
|
||||||
|
applySettings: () => false,
|
||||||
|
resetSettings: () => null,
|
||||||
|
checkOldSettings: () => null,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -154,9 +198,11 @@ const MapRootHandlers = forwardRef(({ children }: WithChildren, fwdRef: Forwarde
|
|||||||
export const MapRootProvider = ({ children, fwdRef, outCommand }: MapRootProviderProps) => {
|
export const MapRootProvider = ({ children, fwdRef, outCommand }: MapRootProviderProps) => {
|
||||||
const { update, ref } = useContextStore<MapRootData>({ ...INITIAL_DATA });
|
const { update, ref } = useContextStore<MapRootData>({ ...INITIAL_DATA });
|
||||||
|
|
||||||
const storedSettings = useMapUserSettings();
|
const storedSettings = useMapUserSettings(ref, outCommand);
|
||||||
|
|
||||||
|
const { windowsSettings, toggleWidgetVisibility, updateWidgetSettings, resetWidgets } =
|
||||||
|
useStoreWidgets(storedSettings);
|
||||||
|
|
||||||
const { windowsSettings, toggleWidgetVisibility, updateWidgetSettings, resetWidgets } = useStoreWidgets();
|
|
||||||
const comments = useComments({ outCommand });
|
const comments = useComments({ outCommand });
|
||||||
const charactersCache = useCharactersCache({ outCommand });
|
const charactersCache = useCharactersCache({ outCommand });
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,18 @@
|
|||||||
import {
|
import {
|
||||||
AvailableThemes,
|
AvailableThemes,
|
||||||
InterfaceStoredSettings,
|
InterfaceStoredSettings,
|
||||||
|
KillsWidgetSettings,
|
||||||
|
LocalWidgetSettings,
|
||||||
MiniMapPlacement,
|
MiniMapPlacement,
|
||||||
|
OnTheMapSettingsType,
|
||||||
PingsPlacement,
|
PingsPlacement,
|
||||||
RoutesType,
|
RoutesType,
|
||||||
} from '@/hooks/Mapper/mapRootProvider/types.ts';
|
} from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import {
|
||||||
|
CURRENT_WINDOWS_VERSION,
|
||||||
|
DEFAULT_WIDGETS,
|
||||||
|
STORED_VISIBLE_WIDGETS_DEFAULT,
|
||||||
|
} from '@/hooks/Mapper/components/mapInterface/constants.tsx';
|
||||||
|
|
||||||
export const STORED_INTERFACE_DEFAULT_VALUES: InterfaceStoredSettings = {
|
export const STORED_INTERFACE_DEFAULT_VALUES: InterfaceStoredSettings = {
|
||||||
isShowMenu: false,
|
isShowMenu: false,
|
||||||
@@ -31,3 +39,29 @@ export const DEFAULT_ROUTES_SETTINGS: RoutesType = {
|
|||||||
avoid_triglavian: false,
|
avoid_triglavian: false,
|
||||||
avoid: [],
|
avoid: [],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const DEFAULT_WIDGET_LOCAL_SETTINGS: LocalWidgetSettings = {
|
||||||
|
compact: true,
|
||||||
|
showOffline: false,
|
||||||
|
version: 0,
|
||||||
|
showShipName: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const DEFAULT_ON_THE_MAP_SETTINGS: OnTheMapSettingsType = {
|
||||||
|
hideOffline: false,
|
||||||
|
version: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const DEFAULT_KILLS_WIDGET_SETTINGS: KillsWidgetSettings = {
|
||||||
|
showAll: false,
|
||||||
|
whOnly: true,
|
||||||
|
excludedSystems: [],
|
||||||
|
version: 2,
|
||||||
|
timeRange: 4,
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getDefaultWidgetProps = () => ({
|
||||||
|
version: CURRENT_WINDOWS_VERSION,
|
||||||
|
visible: STORED_VISIBLE_WIDGETS_DEFAULT,
|
||||||
|
windows: DEFAULT_WIDGETS,
|
||||||
|
});
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
type Settings = Record<string, unknown>;
|
||||||
|
|
||||||
|
export const actualizeSettings = <T extends Settings>(defaultVals: T, vals: T, setVals: (newVals: T) => void) => {
|
||||||
|
let foundNew = false;
|
||||||
|
|
||||||
|
const newVals = Object.keys(defaultVals).reduce((acc, key) => {
|
||||||
|
if (key in acc) {
|
||||||
|
return acc;
|
||||||
|
}
|
||||||
|
|
||||||
|
foundNew = true;
|
||||||
|
|
||||||
|
return {
|
||||||
|
...acc,
|
||||||
|
[key]: defaultVals[key],
|
||||||
|
};
|
||||||
|
}, vals);
|
||||||
|
|
||||||
|
if (foundNew) {
|
||||||
|
setVals(newVals);
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
import { MapUserSettings } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import {
|
||||||
|
DEFAULT_KILLS_WIDGET_SETTINGS,
|
||||||
|
DEFAULT_ON_THE_MAP_SETTINGS,
|
||||||
|
DEFAULT_ROUTES_SETTINGS,
|
||||||
|
DEFAULT_WIDGET_LOCAL_SETTINGS,
|
||||||
|
getDefaultWidgetProps,
|
||||||
|
STORED_INTERFACE_DEFAULT_VALUES,
|
||||||
|
} from '@/hooks/Mapper/mapRootProvider/constants.ts';
|
||||||
|
import { DEFAULT_SIGNATURE_SETTINGS } from '@/hooks/Mapper/constants/signatures.ts';
|
||||||
|
|
||||||
|
// TODO - we need provide and compare version
|
||||||
|
const createWidgetSettingsWithVersion = <T>(settings: T) => {
|
||||||
|
return {
|
||||||
|
version: 0,
|
||||||
|
settings,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createDefaultWidgetSettings = (): MapUserSettings => {
|
||||||
|
return {
|
||||||
|
killsWidget: createWidgetSettingsWithVersion(DEFAULT_KILLS_WIDGET_SETTINGS),
|
||||||
|
localWidget: createWidgetSettingsWithVersion(DEFAULT_WIDGET_LOCAL_SETTINGS),
|
||||||
|
widgets: createWidgetSettingsWithVersion(getDefaultWidgetProps()),
|
||||||
|
routes: createWidgetSettingsWithVersion(DEFAULT_ROUTES_SETTINGS),
|
||||||
|
onTheMap: createWidgetSettingsWithVersion(DEFAULT_ON_THE_MAP_SETTINGS),
|
||||||
|
signaturesWidget: createWidgetSettingsWithVersion(DEFAULT_SIGNATURE_SETTINGS),
|
||||||
|
interface: createWidgetSettingsWithVersion(STORED_INTERFACE_DEFAULT_VALUES),
|
||||||
|
};
|
||||||
|
};
|
||||||
1
assets/js/hooks/Mapper/mapRootProvider/helpers/index.ts
Normal file
1
assets/js/hooks/Mapper/mapRootProvider/helpers/index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './actualizeSettings';
|
||||||
@@ -14,8 +14,8 @@ export const useCommandPings = () => {
|
|||||||
ref.current.update({ pings });
|
ref.current.update({ pings });
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const pingCancelled = useCallback(({ type, solar_system_id }: CommandPingCancelled) => {
|
const pingCancelled = useCallback(({ type, id }: CommandPingCancelled) => {
|
||||||
const newPings = ref.current.pings.filter(x => x.solar_system_id !== solar_system_id && x.type !== type);
|
const newPings = ref.current.pings.filter(x => x.id !== id && x.type !== type);
|
||||||
ref.current.update({ pings: newPings });
|
ref.current.update({ pings: newPings });
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
|||||||
@@ -27,6 +27,7 @@ export const useMapInit = () => {
|
|||||||
main_character_eve_id,
|
main_character_eve_id,
|
||||||
following_character_eve_id,
|
following_character_eve_id,
|
||||||
user_hubs,
|
user_hubs,
|
||||||
|
map_slug,
|
||||||
} = props;
|
} = props;
|
||||||
|
|
||||||
const updateData: Partial<MapRootData> = {};
|
const updateData: Partial<MapRootData> = {};
|
||||||
@@ -98,6 +99,10 @@ export const useMapInit = () => {
|
|||||||
updateData.followingCharacterEveId = following_character_eve_id;
|
updateData.followingCharacterEveId = following_character_eve_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ('map_slug' in props) {
|
||||||
|
updateData.map_slug = map_slug;
|
||||||
|
}
|
||||||
|
|
||||||
update(updateData);
|
update(updateData);
|
||||||
},
|
},
|
||||||
[update, addSystemStatic],
|
[update, addSystemStatic],
|
||||||
|
|||||||
@@ -0,0 +1,66 @@
|
|||||||
|
import { OutCommand, OutCommandHandler } from '@/hooks/Mapper/types';
|
||||||
|
import { Dispatch, SetStateAction, useCallback, useEffect, useRef } from 'react';
|
||||||
|
import {
|
||||||
|
MapUserSettings,
|
||||||
|
MapUserSettingsStructure,
|
||||||
|
RemoteAdminSettingsResponse,
|
||||||
|
} from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
import { createDefaultWidgetSettings } from '@/hooks/Mapper/mapRootProvider/helpers/createDefaultWidgetSettings.ts';
|
||||||
|
import { parseMapUserSettings } from '@/hooks/Mapper/components/helpers';
|
||||||
|
|
||||||
|
interface UseActualizeRemoteMapSettingsProps {
|
||||||
|
outCommand: OutCommandHandler;
|
||||||
|
mapUserSettings: MapUserSettingsStructure;
|
||||||
|
applySettings: (val: MapUserSettings) => void;
|
||||||
|
setMapUserSettings: Dispatch<SetStateAction<MapUserSettingsStructure>>;
|
||||||
|
map_slug: string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useActualizeRemoteMapSettings = ({
|
||||||
|
outCommand,
|
||||||
|
mapUserSettings,
|
||||||
|
setMapUserSettings,
|
||||||
|
applySettings,
|
||||||
|
map_slug,
|
||||||
|
}: UseActualizeRemoteMapSettingsProps) => {
|
||||||
|
const refVars = useRef({ applySettings, mapUserSettings, setMapUserSettings, map_slug });
|
||||||
|
refVars.current = { applySettings, mapUserSettings, setMapUserSettings, map_slug };
|
||||||
|
|
||||||
|
const actualizeRemoteMapSettings = useCallback(async () => {
|
||||||
|
const { applySettings } = refVars.current;
|
||||||
|
|
||||||
|
let res: RemoteAdminSettingsResponse | undefined;
|
||||||
|
try {
|
||||||
|
res = await outCommand({ type: OutCommand.getDefaultSettings, data: null });
|
||||||
|
} catch (error) {
|
||||||
|
// do nothing
|
||||||
|
}
|
||||||
|
|
||||||
|
if (res?.default_settings == null) {
|
||||||
|
applySettings(createDefaultWidgetSettings());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
applySettings(parseMapUserSettings(res.default_settings));
|
||||||
|
} catch (error) {
|
||||||
|
applySettings(createDefaultWidgetSettings());
|
||||||
|
}
|
||||||
|
}, [outCommand]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const { mapUserSettings } = refVars.current;
|
||||||
|
|
||||||
|
// INFO: Do nothing if slug is not set
|
||||||
|
if (map_slug == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// INFO: Do nothing if user have already data
|
||||||
|
if (map_slug in mapUserSettings) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
actualizeRemoteMapSettings();
|
||||||
|
}, [actualizeRemoteMapSettings, map_slug]);
|
||||||
|
};
|
||||||
@@ -1,39 +1,187 @@
|
|||||||
import useLocalStorageState from 'use-local-storage-state';
|
import useLocalStorageState from 'use-local-storage-state';
|
||||||
import { InterfaceStoredSettings, RoutesType } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
import { MapUserSettings, MapUserSettingsStructure } from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
import { DEFAULT_ROUTES_SETTINGS, STORED_INTERFACE_DEFAULT_VALUES } from '@/hooks/Mapper/mapRootProvider/constants.ts';
|
import { useCallback, useEffect, useRef, useState } from 'react';
|
||||||
import { useActualizeSettings } from '@/hooks/Mapper/hooks';
|
import { MapRootData } from '@/hooks/Mapper/mapRootProvider';
|
||||||
import { useEffect } from 'react';
|
import { useSettingsValueAndSetter } from '@/hooks/Mapper/mapRootProvider/hooks/useSettingsValueAndSetter.ts';
|
||||||
import { SESSION_KEY } from '@/hooks/Mapper/constants.ts';
|
import fastDeepEqual from 'fast-deep-equal';
|
||||||
|
import { OutCommandHandler } from '@/hooks/Mapper/types';
|
||||||
|
import { useActualizeRemoteMapSettings } from '@/hooks/Mapper/mapRootProvider/hooks/useActualizeRemoteMapSettings.ts';
|
||||||
|
import { createDefaultWidgetSettings } from '@/hooks/Mapper/mapRootProvider/helpers/createDefaultWidgetSettings.ts';
|
||||||
|
|
||||||
export const useMigrationRoutesSettingsV1 = (update: (upd: RoutesType) => void) => {
|
const EMPTY_OBJ = {};
|
||||||
//TODO if current Date is more than 01.01.2026 - remove this hook.
|
|
||||||
|
|
||||||
useEffect(() => {
|
export const useMapUserSettings = ({ map_slug }: MapRootData, outCommand: OutCommandHandler) => {
|
||||||
const items = localStorage.getItem(SESSION_KEY.routes);
|
const [isReady, setIsReady] = useState(false);
|
||||||
if (items) {
|
const [hasOldSettings, setHasOldSettings] = useState(false);
|
||||||
update(JSON.parse(items));
|
|
||||||
localStorage.removeItem(SESSION_KEY.routes);
|
|
||||||
}
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
}, []);
|
|
||||||
};
|
|
||||||
|
|
||||||
export const useMapUserSettings = () => {
|
const [mapUserSettings, setMapUserSettings] = useLocalStorageState<MapUserSettingsStructure>('map-user-settings', {
|
||||||
const [interfaceSettings, setInterfaceSettings] = useLocalStorageState<InterfaceStoredSettings>(
|
defaultValue: EMPTY_OBJ,
|
||||||
'window:interface:settings',
|
|
||||||
{
|
|
||||||
defaultValue: STORED_INTERFACE_DEFAULT_VALUES,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
const [settingsRoutes, settingsRoutesUpdate] = useLocalStorageState<RoutesType>('window:interface:routes', {
|
|
||||||
defaultValue: DEFAULT_ROUTES_SETTINGS,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
useActualizeSettings(STORED_INTERFACE_DEFAULT_VALUES, interfaceSettings, setInterfaceSettings);
|
const ref = useRef({ mapUserSettings, setMapUserSettings, map_slug });
|
||||||
useActualizeSettings(DEFAULT_ROUTES_SETTINGS, settingsRoutes, settingsRoutesUpdate);
|
ref.current = { mapUserSettings, setMapUserSettings, map_slug };
|
||||||
|
|
||||||
useMigrationRoutesSettingsV1(settingsRoutesUpdate);
|
const applySettings = useCallback((settings: MapUserSettings) => {
|
||||||
|
const { map_slug, mapUserSettings, setMapUserSettings } = ref.current;
|
||||||
|
|
||||||
return { interfaceSettings, setInterfaceSettings, settingsRoutes, settingsRoutesUpdate };
|
if (map_slug == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fastDeepEqual(settings, mapUserSettings[map_slug])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
setMapUserSettings(old => ({
|
||||||
|
...old,
|
||||||
|
[map_slug]: settings,
|
||||||
|
}));
|
||||||
|
return true;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useActualizeRemoteMapSettings({ outCommand, applySettings, mapUserSettings, setMapUserSettings, map_slug });
|
||||||
|
|
||||||
|
const [interfaceSettings, setInterfaceSettings] = useSettingsValueAndSetter(
|
||||||
|
mapUserSettings,
|
||||||
|
setMapUserSettings,
|
||||||
|
map_slug,
|
||||||
|
'interface',
|
||||||
|
);
|
||||||
|
|
||||||
|
const [settingsRoutes, settingsRoutesUpdate] = useSettingsValueAndSetter(
|
||||||
|
mapUserSettings,
|
||||||
|
setMapUserSettings,
|
||||||
|
map_slug,
|
||||||
|
'routes',
|
||||||
|
);
|
||||||
|
|
||||||
|
const [settingsLocal, settingsLocalUpdate] = useSettingsValueAndSetter(
|
||||||
|
mapUserSettings,
|
||||||
|
setMapUserSettings,
|
||||||
|
map_slug,
|
||||||
|
'localWidget',
|
||||||
|
);
|
||||||
|
|
||||||
|
const [settingsSignatures, settingsSignaturesUpdate] = useSettingsValueAndSetter(
|
||||||
|
mapUserSettings,
|
||||||
|
setMapUserSettings,
|
||||||
|
map_slug,
|
||||||
|
'signaturesWidget',
|
||||||
|
);
|
||||||
|
|
||||||
|
const [settingsOnTheMap, settingsOnTheMapUpdate] = useSettingsValueAndSetter(
|
||||||
|
mapUserSettings,
|
||||||
|
setMapUserSettings,
|
||||||
|
map_slug,
|
||||||
|
'onTheMap',
|
||||||
|
);
|
||||||
|
|
||||||
|
const [settingsKills, settingsKillsUpdate] = useSettingsValueAndSetter(
|
||||||
|
mapUserSettings,
|
||||||
|
setMapUserSettings,
|
||||||
|
map_slug,
|
||||||
|
'killsWidget',
|
||||||
|
);
|
||||||
|
|
||||||
|
const [windowsSettings, setWindowsSettings] = useSettingsValueAndSetter(
|
||||||
|
mapUserSettings,
|
||||||
|
setMapUserSettings,
|
||||||
|
map_slug,
|
||||||
|
'widgets',
|
||||||
|
);
|
||||||
|
|
||||||
|
// HERE we MUST work with migrations
|
||||||
|
useEffect(() => {
|
||||||
|
if (isReady) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (map_slug === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mapUserSettings[map_slug] == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO !!!! FROM this date 06.07.2025 - we must work only with migrations
|
||||||
|
// actualizeSettings(STORED_INTERFACE_DEFAULT_VALUES, interfaceSettings, setInterfaceSettings);
|
||||||
|
// actualizeSettings(DEFAULT_ROUTES_SETTINGS, settingsRoutes, settingsRoutesUpdate);
|
||||||
|
// actualizeSettings(DEFAULT_WIDGET_LOCAL_SETTINGS, settingsLocal, settingsLocalUpdate);
|
||||||
|
// actualizeSettings(DEFAULT_SIGNATURE_SETTINGS, settingsSignatures, settingsSignaturesUpdate);
|
||||||
|
// actualizeSettings(DEFAULT_ON_THE_MAP_SETTINGS, settingsOnTheMap, settingsOnTheMapUpdate);
|
||||||
|
// actualizeSettings(DEFAULT_KILLS_WIDGET_SETTINGS, settingsKills, settingsKillsUpdate);
|
||||||
|
|
||||||
|
setIsReady(true);
|
||||||
|
}, [
|
||||||
|
map_slug,
|
||||||
|
mapUserSettings,
|
||||||
|
interfaceSettings,
|
||||||
|
setInterfaceSettings,
|
||||||
|
settingsRoutes,
|
||||||
|
settingsRoutesUpdate,
|
||||||
|
settingsLocal,
|
||||||
|
settingsLocalUpdate,
|
||||||
|
settingsSignatures,
|
||||||
|
settingsSignaturesUpdate,
|
||||||
|
settingsOnTheMap,
|
||||||
|
settingsOnTheMapUpdate,
|
||||||
|
settingsKills,
|
||||||
|
settingsKillsUpdate,
|
||||||
|
isReady,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const checkOldSettings = useCallback(() => {
|
||||||
|
const interfaceSettings = localStorage.getItem('window:interface:settings');
|
||||||
|
const widgetRoutes = localStorage.getItem('window:interface:routes');
|
||||||
|
const widgetLocal = localStorage.getItem('window:interface:local');
|
||||||
|
const widgetKills = localStorage.getItem('kills:widget:settings');
|
||||||
|
const onTheMapOld = localStorage.getItem('window:onTheMap:settings');
|
||||||
|
const widgetsOld = localStorage.getItem('windows:settings:v2');
|
||||||
|
|
||||||
|
setHasOldSettings(!!(widgetsOld || interfaceSettings || widgetRoutes || widgetLocal || widgetKills || onTheMapOld));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
checkOldSettings();
|
||||||
|
}, [checkOldSettings]);
|
||||||
|
|
||||||
|
const getSettingsForExport = useCallback(() => {
|
||||||
|
const { map_slug } = ref.current;
|
||||||
|
|
||||||
|
if (map_slug == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
return JSON.stringify(ref.current.mapUserSettings[map_slug]);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const resetSettings = useCallback(() => {
|
||||||
|
applySettings(createDefaultWidgetSettings());
|
||||||
|
}, [applySettings]);
|
||||||
|
|
||||||
|
return {
|
||||||
|
isReady,
|
||||||
|
hasOldSettings,
|
||||||
|
|
||||||
|
interfaceSettings,
|
||||||
|
setInterfaceSettings,
|
||||||
|
settingsRoutes,
|
||||||
|
settingsRoutesUpdate,
|
||||||
|
settingsLocal,
|
||||||
|
settingsLocalUpdate,
|
||||||
|
settingsSignatures,
|
||||||
|
settingsSignaturesUpdate,
|
||||||
|
settingsOnTheMap,
|
||||||
|
settingsOnTheMapUpdate,
|
||||||
|
settingsKills,
|
||||||
|
settingsKillsUpdate,
|
||||||
|
windowsSettings,
|
||||||
|
setWindowsSettings,
|
||||||
|
|
||||||
|
getSettingsForExport,
|
||||||
|
applySettings,
|
||||||
|
resetSettings,
|
||||||
|
checkOldSettings,
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -0,0 +1,60 @@
|
|||||||
|
import { Dispatch, SetStateAction, useCallback, useMemo, useRef } from 'react';
|
||||||
|
import {
|
||||||
|
MapUserSettings,
|
||||||
|
MapUserSettingsStructure,
|
||||||
|
SettingsWithVersion,
|
||||||
|
} from '@/hooks/Mapper/mapRootProvider/types.ts';
|
||||||
|
|
||||||
|
type ExtractSettings<S extends keyof MapUserSettings> =
|
||||||
|
MapUserSettings[S] extends SettingsWithVersion<infer U> ? U : never;
|
||||||
|
|
||||||
|
type Setter<S extends keyof MapUserSettings> = (
|
||||||
|
value: Partial<ExtractSettings<S>> | ((prev: ExtractSettings<S>) => Partial<ExtractSettings<S>>),
|
||||||
|
) => void;
|
||||||
|
|
||||||
|
type GenerateSettingsReturn<S extends keyof MapUserSettings> = [ExtractSettings<S>, Setter<S>];
|
||||||
|
|
||||||
|
export const useSettingsValueAndSetter = <S extends keyof MapUserSettings>(
|
||||||
|
settings: MapUserSettingsStructure,
|
||||||
|
setSettings: Dispatch<SetStateAction<MapUserSettingsStructure>>,
|
||||||
|
mapId: string | null,
|
||||||
|
setting: S,
|
||||||
|
): GenerateSettingsReturn<S> => {
|
||||||
|
const data = useMemo<ExtractSettings<S>>(() => {
|
||||||
|
if (!mapId) return {} as ExtractSettings<S>;
|
||||||
|
|
||||||
|
const mapSettings = settings[mapId];
|
||||||
|
return (mapSettings?.[setting]?.settings ?? ({} as ExtractSettings<S>)) as ExtractSettings<S>;
|
||||||
|
}, [mapId, setting, settings]);
|
||||||
|
|
||||||
|
const refData = useRef({ mapId, setting, setSettings });
|
||||||
|
refData.current = { mapId, setting, setSettings };
|
||||||
|
|
||||||
|
const setter = useCallback<Setter<S>>(value => {
|
||||||
|
const { mapId, setting, setSettings } = refData.current;
|
||||||
|
|
||||||
|
if (!mapId) return;
|
||||||
|
|
||||||
|
setSettings(all => {
|
||||||
|
const currentMap = all[mapId];
|
||||||
|
const prev = currentMap[setting].settings as ExtractSettings<S>;
|
||||||
|
const version = currentMap[setting].version;
|
||||||
|
|
||||||
|
const patch =
|
||||||
|
typeof value === 'function' ? (value as (p: ExtractSettings<S>) => Partial<ExtractSettings<S>>)(prev) : value;
|
||||||
|
|
||||||
|
return {
|
||||||
|
...all,
|
||||||
|
[mapId]: {
|
||||||
|
...currentMap,
|
||||||
|
[setting]: {
|
||||||
|
version,
|
||||||
|
settings: { ...(prev as any), ...patch } as ExtractSettings<S>,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return [data, setter];
|
||||||
|
};
|
||||||
@@ -1,14 +1,8 @@
|
|||||||
import useLocalStorageState from 'use-local-storage-state';
|
import { DEFAULT_WIDGETS, WidgetsIds } from '@/hooks/Mapper/components/mapInterface/constants.tsx';
|
||||||
import {
|
|
||||||
CURRENT_WINDOWS_VERSION,
|
|
||||||
DEFAULT_WIDGETS,
|
|
||||||
STORED_VISIBLE_WIDGETS_DEFAULT,
|
|
||||||
WidgetsIds,
|
|
||||||
WINDOWS_LOCAL_STORE_KEY,
|
|
||||||
} from '@/hooks/Mapper/components/mapInterface/constants.tsx';
|
|
||||||
import { WindowProps } from '@/hooks/Mapper/components/ui-kit/WindowManager/types.ts';
|
import { WindowProps } from '@/hooks/Mapper/components/ui-kit/WindowManager/types.ts';
|
||||||
import { useCallback, useEffect, useRef } from 'react';
|
import { Dispatch, SetStateAction, useCallback, useRef } from 'react';
|
||||||
import { /*SNAP_GAP,*/ WindowsManagerOnChange } from '@/hooks/Mapper/components/ui-kit/WindowManager';
|
import { WindowsManagerOnChange } from '@/hooks/Mapper/components/ui-kit/WindowManager';
|
||||||
|
import { getDefaultWidgetProps } from '@/hooks/Mapper/mapRootProvider/constants.ts';
|
||||||
|
|
||||||
export type StoredWindowProps = Omit<WindowProps, 'content'>;
|
export type StoredWindowProps = Omit<WindowProps, 'content'>;
|
||||||
export type WindowStoreInfo = {
|
export type WindowStoreInfo = {
|
||||||
@@ -20,17 +14,12 @@ export type WindowStoreInfo = {
|
|||||||
// export type UpdateWidgetSettingsFunc = (widgets: WindowProps[]) => void;
|
// export type UpdateWidgetSettingsFunc = (widgets: WindowProps[]) => void;
|
||||||
export type ToggleWidgetVisibility = (widgetId: WidgetsIds) => void;
|
export type ToggleWidgetVisibility = (widgetId: WidgetsIds) => void;
|
||||||
|
|
||||||
export const getDefaultWidgetProps = () => ({
|
interface UseStoreWidgetsProps {
|
||||||
version: CURRENT_WINDOWS_VERSION,
|
windowsSettings: WindowStoreInfo;
|
||||||
visible: STORED_VISIBLE_WIDGETS_DEFAULT,
|
setWindowsSettings: Dispatch<SetStateAction<WindowStoreInfo>>;
|
||||||
windows: DEFAULT_WIDGETS,
|
}
|
||||||
});
|
|
||||||
|
|
||||||
export const useStoreWidgets = () => {
|
|
||||||
const [windowsSettings, setWindowsSettings] = useLocalStorageState<WindowStoreInfo>(WINDOWS_LOCAL_STORE_KEY, {
|
|
||||||
defaultValue: getDefaultWidgetProps(),
|
|
||||||
});
|
|
||||||
|
|
||||||
|
export const useStoreWidgets = ({ windowsSettings, setWindowsSettings }: UseStoreWidgetsProps) => {
|
||||||
const ref = useRef({ windowsSettings, setWindowsSettings });
|
const ref = useRef({ windowsSettings, setWindowsSettings });
|
||||||
ref.current = { windowsSettings, setWindowsSettings };
|
ref.current = { windowsSettings, setWindowsSettings };
|
||||||
|
|
||||||
@@ -83,33 +72,6 @@ export const useStoreWidgets = () => {
|
|||||||
});
|
});
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
const { setWindowsSettings } = ref.current;
|
|
||||||
|
|
||||||
const raw = localStorage.getItem(WINDOWS_LOCAL_STORE_KEY);
|
|
||||||
if (!raw) {
|
|
||||||
console.warn('No windows found in local storage!!');
|
|
||||||
|
|
||||||
setWindowsSettings(getDefaultWidgetProps());
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const { version, windows, visible, viewPort } = JSON.parse(raw) as WindowStoreInfo;
|
|
||||||
if (!version || CURRENT_WINDOWS_VERSION > version) {
|
|
||||||
setWindowsSettings(getDefaultWidgetProps());
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line no-debugger
|
|
||||||
const out = windows.filter(x => DEFAULT_WIDGETS.find(def => def.id === x.id));
|
|
||||||
|
|
||||||
setWindowsSettings({
|
|
||||||
version: CURRENT_WINDOWS_VERSION,
|
|
||||||
windows: out as WindowProps[],
|
|
||||||
visible,
|
|
||||||
viewPort,
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const resetWidgets = useCallback(() => ref.current.setWindowsSettings(getDefaultWidgetProps()), []);
|
const resetWidgets = useCallback(() => ref.current.setWindowsSettings(getDefaultWidgetProps()), []);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
|||||||
@@ -1,3 +1,6 @@
|
|||||||
|
import { WindowStoreInfo } from '@/hooks/Mapper/mapRootProvider/hooks/useStoreWidgets.ts';
|
||||||
|
import { SignatureSettingsType } from '@/hooks/Mapper/constants/signatures.ts';
|
||||||
|
|
||||||
export enum AvailableThemes {
|
export enum AvailableThemes {
|
||||||
default = 'default',
|
default = 'default',
|
||||||
pathfinder = 'pathfinder',
|
pathfinder = 'pathfinder',
|
||||||
@@ -43,3 +46,46 @@ export type RoutesType = {
|
|||||||
avoid_triglavian: boolean;
|
avoid_triglavian: boolean;
|
||||||
avoid: number[];
|
avoid: number[];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type LocalWidgetSettings = {
|
||||||
|
compact: boolean;
|
||||||
|
showOffline: boolean;
|
||||||
|
version: number;
|
||||||
|
showShipName: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type OnTheMapSettingsType = {
|
||||||
|
hideOffline: boolean;
|
||||||
|
version: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type KillsWidgetSettings = {
|
||||||
|
showAll: boolean;
|
||||||
|
whOnly: boolean;
|
||||||
|
excludedSystems: number[];
|
||||||
|
version: number;
|
||||||
|
timeRange: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type SettingsWithVersion<T> = {
|
||||||
|
version: number;
|
||||||
|
settings: T;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type MapUserSettings = {
|
||||||
|
widgets: SettingsWithVersion<WindowStoreInfo>;
|
||||||
|
interface: SettingsWithVersion<InterfaceStoredSettings>;
|
||||||
|
onTheMap: SettingsWithVersion<OnTheMapSettingsType>;
|
||||||
|
routes: SettingsWithVersion<RoutesType>;
|
||||||
|
localWidget: SettingsWithVersion<LocalWidgetSettings>;
|
||||||
|
signaturesWidget: SettingsWithVersion<SignatureSettingsType>;
|
||||||
|
killsWidget: SettingsWithVersion<KillsWidgetSettings>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type MapUserSettingsStructure = {
|
||||||
|
[mapId: string]: MapUserSettings;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type WdResponse<T> = T;
|
||||||
|
|
||||||
|
export type RemoteAdminSettingsResponse = { default_settings?: string };
|
||||||
|
|||||||
@@ -97,6 +97,7 @@ export type CommandInit = {
|
|||||||
is_subscription_active?: boolean;
|
is_subscription_active?: boolean;
|
||||||
main_character_eve_id?: string | null;
|
main_character_eve_id?: string | null;
|
||||||
following_character_eve_id?: string | null;
|
following_character_eve_id?: string | null;
|
||||||
|
map_slug?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type CommandAddSystems = SolarSystemRawType[];
|
export type CommandAddSystems = SolarSystemRawType[];
|
||||||
@@ -150,7 +151,7 @@ export type CommandUpdateTracking = {
|
|||||||
follow: boolean;
|
follow: boolean;
|
||||||
};
|
};
|
||||||
export type CommandPingAdded = PingData[];
|
export type CommandPingAdded = PingData[];
|
||||||
export type CommandPingCancelled = Pick<PingData, 'type' | 'solar_system_id'>;
|
export type CommandPingCancelled = Pick<PingData, 'type' | 'id'>;
|
||||||
|
|
||||||
export interface UserSettings {
|
export interface UserSettings {
|
||||||
primaryCharacterId?: string;
|
primaryCharacterId?: string;
|
||||||
@@ -268,6 +269,8 @@ export enum OutCommand {
|
|||||||
showTracking = 'show_tracking',
|
showTracking = 'show_tracking',
|
||||||
getUserSettings = 'get_user_settings',
|
getUserSettings = 'get_user_settings',
|
||||||
updateUserSettings = 'update_user_settings',
|
updateUserSettings = 'update_user_settings',
|
||||||
|
saveDefaultSettings = 'save_default_settings',
|
||||||
|
getDefaultSettings = 'get_default_settings',
|
||||||
unlinkSignature = 'unlink_signature',
|
unlinkSignature = 'unlink_signature',
|
||||||
searchSystems = 'search_systems',
|
searchSystems = 'search_systems',
|
||||||
undoDeleteSignatures = 'undo_delete_signatures',
|
undoDeleteSignatures = 'undo_delete_signatures',
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ export enum PingType {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export type PingData = {
|
export type PingData = {
|
||||||
|
id: string;
|
||||||
inserted_at: number;
|
inserted_at: number;
|
||||||
character_eve_id: string;
|
character_eve_id: string;
|
||||||
solar_system_id: string;
|
solar_system_id: string;
|
||||||
|
|||||||
@@ -1,19 +1,69 @@
|
|||||||
import { MapHandlers } from '@/hooks/Mapper/types/mapHandlers.ts';
|
import { MapHandlers } from '@/hooks/Mapper/types/mapHandlers.ts';
|
||||||
import { RefObject, useCallback } from 'react';
|
import { RefObject, useCallback, useEffect, useRef } from 'react';
|
||||||
|
import debounce from 'lodash.debounce';
|
||||||
|
import usePageVisibility from '@/hooks/Mapper/hooks/usePageVisibility.ts';
|
||||||
|
|
||||||
|
// const inIndex = 0;
|
||||||
|
// const prevEventTime = +new Date();
|
||||||
|
const LAST_VERSION_KEY = 'wandererLastVersion';
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
export const useMapperHandlers = (handlerRefs: RefObject<MapHandlers>[], hooksRef: RefObject<any>) => {
|
export const useMapperHandlers = (handlerRefs: RefObject<MapHandlers>[], hooksRef: RefObject<any>) => {
|
||||||
|
const visible = usePageVisibility();
|
||||||
|
const wasHiddenOnce = useRef(false);
|
||||||
|
const visibleRef = useRef(visible);
|
||||||
|
visibleRef.current = visible;
|
||||||
|
|
||||||
|
// TODO - do not delete THIS code it needs for debug
|
||||||
|
// const [record, setRecord] = useLocalStorageState<boolean>('record', {
|
||||||
|
// defaultValue: false,
|
||||||
|
// });
|
||||||
|
// const [recordsList, setRecordsList] = useLocalStorageState<{ type; data }[]>('recordsList', {
|
||||||
|
// defaultValue: [],
|
||||||
|
// });
|
||||||
|
//
|
||||||
|
// const ref = useRef({ record, setRecord, recordsList, setRecordsList });
|
||||||
|
// ref.current = { record, setRecord, recordsList, setRecordsList };
|
||||||
|
//
|
||||||
|
// const recordBufferRef = useRef<{ type; data }[]>([]);
|
||||||
|
// useEffect(() => {
|
||||||
|
// if (record || recordBufferRef.current.length === 0) {
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// ref.current.setRecordsList([...recordBufferRef.current]);
|
||||||
|
// recordBufferRef.current = [];
|
||||||
|
// }, [record]);
|
||||||
|
|
||||||
const handleCommand = useCallback(
|
const handleCommand = useCallback(
|
||||||
|
// @ts-ignore
|
||||||
async ({ type, data }) => {
|
async ({ type, data }) => {
|
||||||
if (!hooksRef.current) {
|
if (!hooksRef.current) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO - do not delete THIS code it needs for debug
|
||||||
|
// console.log('JOipP', `OUT`, ref.current.record, { type, data });
|
||||||
|
// if (ref.current.record) {
|
||||||
|
// recordBufferRef.current.push({ type, data });
|
||||||
|
// }
|
||||||
|
|
||||||
|
// 'ui_loaded'
|
||||||
return await hooksRef.current.pushEventAsync(type, data);
|
return await hooksRef.current.pushEventAsync(type, data);
|
||||||
},
|
},
|
||||||
[hooksRef.current],
|
[hooksRef.current],
|
||||||
);
|
);
|
||||||
|
|
||||||
const handleMapEvent = useCallback(({ type, body }) => {
|
// @ts-ignore
|
||||||
|
const eventsBufferRef = useRef<{ type; body }[]>([]);
|
||||||
|
|
||||||
|
const eventTick = useCallback(
|
||||||
|
debounce(() => {
|
||||||
|
if (eventsBufferRef.current.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { type, body } = eventsBufferRef.current.shift()!;
|
||||||
handlerRefs.forEach(ref => {
|
handlerRefs.forEach(ref => {
|
||||||
if (!ref.current) {
|
if (!ref.current) {
|
||||||
return;
|
return;
|
||||||
@@ -21,7 +71,51 @@ export const useMapperHandlers = (handlerRefs: RefObject<MapHandlers>[], hooksRe
|
|||||||
|
|
||||||
ref.current?.command(type, body);
|
ref.current?.command(type, body);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// TODO - do not delete THIS code it needs for debug
|
||||||
|
// console.log('JOipP', `Tick Buff`, eventsBufferRef.current.length);
|
||||||
|
|
||||||
|
if (eventsBufferRef.current.length > 0) {
|
||||||
|
eventTick();
|
||||||
|
}
|
||||||
|
}, 10),
|
||||||
|
[],
|
||||||
|
);
|
||||||
|
const eventTickRef = useRef(eventTick);
|
||||||
|
eventTickRef.current = eventTick;
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
const handleMapEvent = useCallback(({ type, body }) => {
|
||||||
|
// TODO - do not delete THIS code it needs for debug
|
||||||
|
// const currentTime = +new Date();
|
||||||
|
// const timeDiff = currentTime - prevEventTime;
|
||||||
|
// prevEventTime = currentTime;
|
||||||
|
// console.log('JOipP', `IN [${inIndex++}] [${timeDiff}] ${getFormattedTime()}`, { type, body });
|
||||||
|
|
||||||
|
if (!eventTickRef.current || !visibleRef.current) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
eventsBufferRef.current.push({ type, body });
|
||||||
|
eventTickRef.current();
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!visible && !wasHiddenOnce.current) {
|
||||||
|
wasHiddenOnce.current = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!wasHiddenOnce.current) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!visible) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
hooksRef.current.pushEventAsync('ui_loaded', { version: localStorage.getItem(LAST_VERSION_KEY) });
|
||||||
|
}, [hooksRef.current, visible]);
|
||||||
|
|
||||||
return { handleCommand, handleMapEvent };
|
return { handleCommand, handleMapEvent };
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,2 +1,4 @@
|
|||||||
export * from './contextStore';
|
export * from './contextStore';
|
||||||
export * from './getQueryVariable';
|
export * from './getQueryVariable';
|
||||||
|
export * from './loadTextFile';
|
||||||
|
export * from './saveToFile';
|
||||||
|
|||||||
27
assets/js/hooks/Mapper/utils/loadTextFile.ts
Normal file
27
assets/js/hooks/Mapper/utils/loadTextFile.ts
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
export function loadTextFile(): Promise<string> {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const input = document.createElement('input');
|
||||||
|
input.type = 'file';
|
||||||
|
input.accept = 'application/json,.json';
|
||||||
|
|
||||||
|
input.onchange = () => {
|
||||||
|
const file = input.files?.[0];
|
||||||
|
if (!file) {
|
||||||
|
reject(new Error('No file selected'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onload = () => {
|
||||||
|
resolve(reader.result as string);
|
||||||
|
};
|
||||||
|
reader.onerror = () => {
|
||||||
|
reject(reader.error);
|
||||||
|
};
|
||||||
|
|
||||||
|
reader.readAsText(file);
|
||||||
|
};
|
||||||
|
|
||||||
|
input.click();
|
||||||
|
});
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user