Compare commits

...

28 Commits

Author SHA1 Message Date
CI
f767e42e6f chore: release version v1.2.5 2024-10-04 21:56:47 +00:00
Dmitry Popov
3051eb6369 Merge branch 'main' of github.com:wanderer-industries/wanderer 2024-10-05 01:56:19 +04:00
Dmitry Popov
a41faddca3 fix(Core): Add system "true security" correction 2024-10-05 01:56:16 +04:00
CI
469038730e chore: release version v1.2.4 2024-10-03 09:27:53 +00:00
Dmitry Popov
b1fe5d2453 fix(Map): Remove duplicate connections 2024-10-03 13:27:21 +04:00
CI
f43e717da0 chore: release version v1.2.3 2024-10-02 17:52:44 +00:00
Dmitry Popov
95c8d4eef8 Merge branch 'main' of github.com:wanderer-industries/wanderer 2024-10-02 21:52:08 +04:00
Dmitry Popov
747ca0ee82 fix(Map): Fix map loading after select a different map. 2024-10-02 21:52:04 +04:00
CI
35a0184ec3 chore: release version v1.2.2 2024-10-02 12:50:16 +00:00
Dmitry Popov
96e1e5328c Merge branch 'main' of github.com:wanderer-industries/wanderer 2024-10-02 16:49:27 +04:00
Dmitry Popov
7f98d6a0d8 chore: release version v1.2.0 2024-10-02 16:49:23 +04:00
CI
0194e25696 chore: release version v1.2.1 2024-10-02 12:48:06 +00:00
Dmitry Popov
189442e50f Merge branch 'main' of github.com:wanderer-industries/wanderer 2024-10-02 16:47:34 +04:00
Dmitry Popov
d9bed070ec fix(ACL): Fix allowing to save map/access list with empty owner set 2024-10-02 16:47:29 +04:00
CI
a6193da8b5 chore: release version v1.2.0 2024-09-29 19:01:16 +00:00
Aleksei Chichenkov
50d35b207d Merge pull request #14 from wanderer-industries/feat-wnd-13
feat(Map): Add ability to open jump planner from routes
2024-09-29 22:00:44 +03:00
achichenkov
19eb45bfa1 feat(Map): Add ability to open jump planner from routes
Fixes #13
2024-09-29 21:47:35 +03:00
CI
01e0b24d9d chore: release version v1.1.0 2024-09-29 15:07:13 +00:00
Aleksei Chichenkov
3c8024b16c Merge pull request #12 from wanderer-industries/feat-wnd-11
feat(Map): Add highlighting for imperial space systems depends on fac…
2024-09-29 18:06:41 +03:00
achichenkov
4c0ad0dd66 feat(Map): Add highlighting for imperial space systems depends on faction
Fixes #11
2024-09-29 17:57:07 +03:00
CI
501840086b chore: release version v1.0.23 2024-09-25 09:52:47 +00:00
Dmitry Popov
240b180857 Merge branch 'main' of github.com:wanderer-industries/wanderer 2024-09-25 13:52:14 +04:00
Dmitry Popov
2bc5d0aaea fix(Map): Main map doesn't load back after refreshing/switching pages
fixes #8
2024-09-25 13:52:10 +04:00
CI
df66aa79b8 chore: release version v1.0.22 2024-09-25 08:24:50 +00:00
Dmitry Popov
6ea6a59ce3 fix(Map): Main map doesn't load back after refreshing/switching pages
fixes #8
2024-09-25 12:24:14 +04:00
CI
f3afa4d9d2 chore: release version v1.0.21 2024-09-24 20:21:54 +00:00
Dmitry Popov
e33d81eda1 Merge branch 'main' of github.com:wanderer-industries/wanderer 2024-09-25 00:21:17 +04:00
Dmitry Popov
0fc4863dc4 fix(Map): Main map doesn't load back after refreshing/switching pages
fixes #8
2024-09-25 00:21:14 +04:00
47 changed files with 888 additions and 448 deletions

2
.gitignore vendored
View File

@@ -12,6 +12,8 @@
# Ignore assets that are produced by build tools.
/priv/static/assets/
/priv/static/icons/
/priv/static/images/
/priv/static/*.js
/priv/static/*.css

View File

@@ -2,44 +2,107 @@
<!-- changelog -->
## [v1.0.20](https://github.com/wanderer-industries/wanderer/compare/v1.0.19...v1.0.20) (2024-09-23)
## [v1.2.5](https://github.com/wanderer-industries/wanderer/compare/v1.2.4...v1.2.5) (2024-10-04)
### Bug Fixes:
* Core: Add system "true security" correction
## [v1.2.4](https://github.com/wanderer-industries/wanderer/compare/v1.2.3...v1.2.4) (2024-10-03)
### Bug Fixes:
* Map: Remove duplicate connections
## [v1.2.3](https://github.com/wanderer-industries/wanderer/compare/v1.2.2...v1.2.3) (2024-10-02)
### Bug Fixes:
* Map: Fix map loading after select a different map.
## [v1.2.2](https://github.com/wanderer-industries/wanderer/compare/v1.2.1...v1.2.2) (2024-10-02)
## [v1.2.1](https://github.com/wanderer-industries/wanderer/compare/v1.2.0...v1.2.1) (2024-10-02)
### Bug Fixes:
* ACL: Fix allowing to save map/access list with empty owner set
## [v1.2.0](https://github.com/wanderer-industries/wanderer/compare/v1.1.0...v1.2.0) (2024-09-29)
### Features:
* Map: Add ability to open jump planner from routes
## [v1.1.0](https://github.com/wanderer-industries/wanderer/compare/v1.0.23...v1.1.0) (2024-09-29)
### Features:
* Map: Add highlighting for imperial space systems depends on faction
## [v1.0.23](https://github.com/wanderer-industries/wanderer/compare/v1.0.22...v1.0.23) (2024-09-25)
### Bug Fixes:
* Map: Main map doesn't load back after refreshing/switching pages
## [v1.0.22](https://github.com/wanderer-industries/wanderer/compare/v1.0.21...v1.0.22) (2024-09-25)
### Bug Fixes
* Map: Main map doesn't load back after refreshing/switching pages
## [v1.0.21](https://github.com/wanderer-industries/wanderer/compare/v1.0.20...v1.0.21) (2024-09-24)
### Bug Fixes
* Map: Main map doesn't load back after refreshing/switching pages
## [v1.0.20](https://github.com/wanderer-industries/wanderer/compare/v1.0.19...v1.0.20) (2024-09-23)
### Bug Fixes
* core: Small fixes & improvements
## [v1.0.19](https://github.com/wanderer-industries/wanderer/compare/v1.0.18...v1.0.19) (2024-09-23)
### Bug Fixes:
### Bug Fixes
* ACL: Fix adding empty members list
## [v1.0.18](https://github.com/wanderer-industries/wanderer/compare/v1.0.17...v1.0.18) (2024-09-22)
### Bug Fixes:
### Bug Fixes
* ACL: Cant delete ACL list after map deletion #5
## [v1.0.17](https://github.com/wanderer-industries/wanderer/compare/v1.0.16...v1.0.17) (2024-09-21)
## [v1.0.16](https://github.com/wanderer-industries/wanderer/compare/v1.0.15...v1.0.16) (2024-09-21)
### Bug Fixes:
### Bug Fixes
* Map: commented console log
@@ -49,103 +112,58 @@
## [v1.0.15](https://github.com/wanderer-industries/wanderer/compare/v1.0.14...v1.0.15) (2024-09-21)
### Bug Fixes:
### Bug Fixes
* map: Show a proper user notification if map was deleted/archived
## [v1.0.14](https://github.com/wanderer-industries/wanderer/compare/v1.0.13...v1.0.14) (2024-09-21)
## [v1.0.13](https://github.com/wanderer-industries/wanderer/compare/v1.0.12...v1.0.13) (2024-09-21)
### Bug Fixes:
### Bug Fixes
* tracking: Ensure user has at least one character tracked to work with map
## [v1.0.12](https://github.com/wanderer-industries/wanderer/compare/v1.0.11...v1.0.12) (2024-09-20)
### Bug Fixes:
### Bug Fixes
* audit: Hide character for non-character map activities
## [v1.0.11](https://github.com/wanderer-industries/wanderer/compare/v1.0.10...v1.0.11) (2024-09-20)
## [v1.0.10](https://github.com/wanderer-industries/wanderer/compare/v1.0.9...v1.0.10) (2024-09-19)
### Bug Fixes:
### Bug Fixes
* signatures: Fix update signatures error if no character tracked on map
## [v1.0.9](https://github.com/wanderer-industries/wanderer/compare/v1.0.8...v1.0.9) (2024-09-19)
### Bug Fixes:
### Bug Fixes
* core: Fix system add error if it's already added on map
## [v1.0.8](https://github.com/wanderer-industries/wanderer/compare/v1.0.7...v1.0.8) (2024-09-19)
### Bug Fixes:
### Bug Fixes
* docker: Fix DB connection in docker-compose internal network
## [v1.0.7](https://github.com/wanderer-industries/wanderer/compare/v1.0.6...v1.0.7) (2024-09-19)
## [v1.0.6](https://github.com/wanderer-industries/wanderer/compare/v1.0.5...v1.0.6) (2024-09-18)
## [v1.0.5](https://github.com/wanderer-industries/wanderer/compare/v1.0.4...v1.0.5) (2024-09-18)
## [v1.0.4](https://github.com/wanderer-industries/wanderer/compare/v1.0.3...v1.0.4) (2024-09-18)
### Bug Fixes:
### Bug Fixes
* core: skip search results for failed character info request
## [v1.0.3](https://github.com/wanderer-industries/wanderer/compare/v1.0.2...v1.0.3) (2024-09-18)
## [v1.0.2](https://github.com/wanderer-industries/wanderer/compare/v1.0.1...v1.0.2) (2024-09-18)
## [v1.0.1](https://github.com/wanderer-industries/wanderer/compare/v1.0.0...v1.0.1) (2024-09-18)

View File

@@ -8,17 +8,21 @@ import { getSystemById } from '@/hooks/Mapper/helpers';
import { useWaypointMenu } from '@/hooks/Mapper/components/contexts/hooks';
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
import { FastSystemActions } from '@/hooks/Mapper/components/contexts/components';
import { useJumpPlannerMenu } from '@/hooks/Mapper/components/contexts/hooks/useJumpPlannerMenu';
import { Route } from '@/hooks/Mapper/types/routes.ts';
export interface ContextMenuSystemInfoProps {
systemStatics: Map<number, SolarSystemStaticInfoRaw>;
hubs: string[];
contextMenuRef: RefObject<ContextMenu>;
systemId: string | undefined;
systemIdFrom?: string | undefined;
systems: SolarSystemRawType[];
onOpenSettings(): void;
onHubToggle(): void;
onAddSystem(): void;
onWaypointSet: WaypointSetContextHandler;
routes: Route[];
}
export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
@@ -30,9 +34,12 @@ export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
onAddSystem,
onWaypointSet,
systemId,
systemIdFrom,
hubs,
routes,
}) => {
const getWaypointMenu = useWaypointMenu(onWaypointSet);
const getJumpPlannerMenu = useJumpPlannerMenu(systems, systemIdFrom);
const items: MenuItem[] = useMemo(() => {
const system = systemId ? systemStatics.get(parseInt(systemId)) : undefined;
@@ -55,7 +62,9 @@ export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
);
},
},
{ separator: true },
...getJumpPlannerMenu(system, routes),
...getWaypointMenu(systemId, system.system_class),
{
label: !hubs.includes(systemId) ? 'Add in Routes' : 'Remove from Routes',
@@ -72,7 +81,17 @@ export const ContextMenuSystemInfo: React.FC<ContextMenuSystemInfoProps> = ({
]
: []),
];
}, [systemId, systemStatics, systems, getWaypointMenu, hubs, onHubToggle, onAddSystem, onOpenSettings]);
}, [
systemId,
systemStatics,
systems,
getJumpPlannerMenu,
getWaypointMenu,
hubs,
onHubToggle,
onAddSystem,
onOpenSettings,
]);
return (
<>

View File

@@ -4,6 +4,7 @@ import { Commands, MapHandlers, OutCommand, OutCommandHandler } from '@/hooks/Ma
import { WaypointSetContextHandler } from '@/hooks/Mapper/components/contexts/types.ts';
import { ctxManager } from '@/hooks/Mapper/utils/contextManager.ts';
import * as React from 'react';
import { SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
interface UseContextMenuSystemHandlersProps {
hubs: string[];
@@ -15,16 +16,21 @@ export const useContextMenuSystemInfoHandlers = ({ hubs, outCommand, mapRef }: U
const contextMenuRef = useRef<ContextMenu | null>(null);
const [system, setSystem] = useState<string>();
const routeRef = useRef<(SolarSystemStaticInfoRaw | undefined)[]>([]);
const ref = useRef({ hubs, system, outCommand, mapRef });
ref.current = { hubs, system, outCommand, mapRef };
const open = useCallback((ev: React.SyntheticEvent, systemId: string) => {
setSystem(systemId);
ev.preventDefault();
ctxManager.next('ctxSysInfo', contextMenuRef.current);
contextMenuRef.current?.show(ev);
}, []);
const open = useCallback(
(ev: React.SyntheticEvent, systemId: string, route: (SolarSystemStaticInfoRaw | undefined)[]) => {
setSystem(systemId);
routeRef.current = route;
ev.preventDefault();
ctxManager.next('ctxSysInfo', contextMenuRef.current);
contextMenuRef.current?.show(ev);
},
[],
);
const onHubToggle = useCallback(() => {
const { hubs, system, outCommand } = ref.current;

View File

@@ -0,0 +1 @@
export * from './useJumpPlannerMenu.tsx';

View File

@@ -0,0 +1,129 @@
import { MenuItem } from 'primereact/menuitem';
import { PrimeIcons } from 'primereact/api';
import { useCallback } from 'react';
import { isPossibleSpace } from '@/hooks/Mapper/components/map/helpers/isKnownSpace.ts';
import { Route } from '@/hooks/Mapper/types/routes.ts';
import { SolarSystemRawType, SolarSystemStaticInfoRaw } from '@/hooks/Mapper/types';
import { getSystemById } from '@/hooks/Mapper/helpers';
import { SOLAR_SYSTEM_CLASS_IDS } from '@/hooks/Mapper/components/map/constants.ts';
const imperialSpace = [SOLAR_SYSTEM_CLASS_IDS.hs, SOLAR_SYSTEM_CLASS_IDS.ls, SOLAR_SYSTEM_CLASS_IDS.ns];
const criminalSpace = [SOLAR_SYSTEM_CLASS_IDS.ls, SOLAR_SYSTEM_CLASS_IDS.ns];
enum JUMP_SHIP_TYPE {
BLACK_OPS = 'Marshal',
JUMP_FREIGHTER = 'Anshar',
RORQUAL = 'Rorqual',
CAPITAL = 'Thanatos',
SUPER_CAPITAL = 'Avatar',
}
export const openJumpPlan = (jumpShipType: JUMP_SHIP_TYPE, from: string, to: string) => {
return window.open(`https://evemaps.dotlan.net/jump/${jumpShipType},544/${from}:${to}`, '_blank');
};
const BRACKET_ICONS = {
npcsuperCarrier_32: '/icons/brackets/npcsuperCarrier_32.png',
carrier_32: '/icons/brackets/carrier_32.png',
battleship_32: '/icons/brackets/battleship_32.png',
freighter_32: '/icons/brackets/freighter_32.png',
};
const renderIcon = (icon: string) => {
return (
<div className="flex justify-center items-center mr-1.5 pt-px">
<img src={icon} style={{ width: 20, height: 20 }} />
</div>
);
};
export const useJumpPlannerMenu = (
systems: SolarSystemRawType[],
systemIdFrom?: string | undefined,
): ((systemId: SolarSystemStaticInfoRaw, routes: Route[]) => MenuItem[]) => {
return useCallback(
(destination: SolarSystemStaticInfoRaw) => {
if (!destination || !systemIdFrom) {
return [];
}
const origin = getSystemById(systems, systemIdFrom)?.system_static_info;
if (!origin) {
return [];
}
const isShowBOorJumpFreighter =
isPossibleSpace(imperialSpace, origin.system_class) && isPossibleSpace(criminalSpace, destination.system_class);
const isShowCapital =
isPossibleSpace(criminalSpace, origin.system_class) && isPossibleSpace(criminalSpace, destination.system_class);
if (!isShowBOorJumpFreighter && !isShowCapital) {
return [];
}
return [
{
label: 'In Jump Planner',
icon: PrimeIcons.SEND,
items: [
...(isShowBOorJumpFreighter
? [
{
label: 'Black Ops',
icon: renderIcon(BRACKET_ICONS.battleship_32),
command: () => {
openJumpPlan(JUMP_SHIP_TYPE.BLACK_OPS, origin.solar_system_name, destination.solar_system_name);
},
},
{
label: 'Jump Freighter',
icon: renderIcon(BRACKET_ICONS.freighter_32),
command: () => {
openJumpPlan(
JUMP_SHIP_TYPE.JUMP_FREIGHTER,
origin.solar_system_name,
destination.solar_system_name,
);
},
},
{
label: 'Rorqual',
icon: renderIcon(BRACKET_ICONS.freighter_32),
command: () => {
openJumpPlan(JUMP_SHIP_TYPE.RORQUAL, origin.solar_system_name, destination.solar_system_name);
},
},
]
: []),
...(isShowCapital
? [
{
label: 'Capital',
icon: renderIcon(BRACKET_ICONS.carrier_32),
command: () => {
openJumpPlan(JUMP_SHIP_TYPE.CAPITAL, origin.solar_system_name, destination.solar_system_name);
},
},
{
label: 'Super Capital',
icon: renderIcon(BRACKET_ICONS.npcsuperCarrier_32),
command: () => {
openJumpPlan(
JUMP_SHIP_TYPE.SUPER_CAPITAL,
origin.solar_system_name,
destination.solar_system_name,
);
},
},
]
: []),
],
},
];
},
[systems, systemIdFrom],
);
};

View File

@@ -1,4 +1,4 @@
import React, { ForwardedRef, forwardRef, MouseEvent, useCallback } from 'react';
import { ForwardedRef, forwardRef, MouseEvent, useCallback, useEffect } from 'react';
import ReactFlow, {
Background,
ConnectionMode,
@@ -94,6 +94,7 @@ interface MapCompProps {
minimapClasses?: string;
isShowMinimap?: boolean;
onSystemContextMenu: (event: MouseEvent<Element>, systemId: string) => void;
showKSpaceBG?: boolean;
}
const MapComp = ({
@@ -105,6 +106,7 @@ const MapComp = ({
onConnectionInfoClick,
onSelectionContextMenu,
isShowMinimap,
showKSpaceBG,
}: MapCompProps) => {
const [nodes, , onNodesChange] = useNodesState<SolarSystemRawType>(initialNodes);
const [edges, , onEdgesChange] = useEdgesState<Edge<SolarSystemConnection>[]>(initialEdges);
@@ -169,6 +171,13 @@ const MapComp = ({
localStorage.setItem(SESSION_KEY.viewPort, JSON.stringify(viewport));
};
useEffect(() => {
update(x => ({
...x,
showKSpaceBG: showKSpaceBG,
}));
}, [showKSpaceBG, update]);
return (
<>
<div className={classes.MapRoot}>

View File

@@ -7,6 +7,7 @@ export type MapData = MapUnionTypes & {
isConnecting: boolean;
hoverNodeId: string | null;
visibleNodes: Set<string>;
showKSpaceBG: boolean;
};
interface MapProviderProps {
@@ -27,6 +28,7 @@ const INITIAL_DATA: MapData = {
connections: [],
hoverNodeId: null,
visibleNodes: new Set(),
showKSpaceBG: false,
};
export interface MapContextProps {
@@ -38,6 +40,7 @@ export interface MapContextProps {
const MapContext = createContext<MapContextProps>({
update: () => {},
data: { ...INITIAL_DATA },
// @ts-ignore
outCommand: async () => void 0,
});

View File

@@ -23,6 +23,62 @@ $tooltip-bg: #202020; // Темный фон для подсказок
border-radius: 5px;
position: relative;
z-index: 1;
overflow: hidden;
&.Mataria, &.Amarria, &.Gallente, &.Caldaria {
&::Before {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-size: cover;
background-position: 50% 50%;
z-index: -1;
background-repeat: no-repeat;
border-radius: 3px;
}
}
&.Mataria {
&::before {
background-image: url("/images/mataria.png");
opacity: 0.6;
background-position-x: -28px;
background-position-y: -3px;
}
}
&.Caldaria {
&::before {
background-image: url("/images/caldaria.png");
opacity: 0.6;
background-position-x: -16px;
background-position-y: -17px;
}
}
&.Amarria {
&::before {
opacity: 0.45;
background-image: url("/images/amarr.png");
background-position-x: 0px;
background-position-y: -1px;
width: calc(100% + 10px)
}
}
&.Gallente {
&::before {
opacity: 0.6;
background-image: url("/images/gallente.png");
background-position-x: -1px;
background-position-y: -10px;
}
}
&.selected {
border-color: $pastel-pink;

View File

@@ -19,6 +19,14 @@ import { PrimeIcons } from 'primereact/api';
import { LabelsManager } from '@/hooks/Mapper/utils/labelsManager.ts';
import { OutCommand } from '@/hooks/Mapper/types';
import { useDoubleClick } from '@/hooks/Mapper/hooks/useDoubleClick.ts';
import { REGIONS_MAP, Spaces } from '@/hooks/Mapper/constants';
const SpaceToClass: Record<string, string> = {
[Spaces.Caldari]: classes.Caldaria,
[Spaces.Matar]: classes.Mataria,
[Spaces.Amarr]: classes.Amarria,
[Spaces.Gallente]: classes.Gallente,
};
const sortedLabels = (labels: string[]) => {
if (labels === null) {
@@ -50,6 +58,7 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
statics,
effect_name,
region_name,
region_id,
is_shattered,
solar_system_name,
} = data.system_static_info;
@@ -69,6 +78,7 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
isConnecting,
hoverNodeId,
visibleNodes,
showKSpaceBG,
},
outCommand,
} = useMapState();
@@ -114,6 +124,9 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
const showHandlers = isConnecting || hoverNodeId === id;
const space = showKSpaceBG ? REGIONS_MAP[region_id] : '';
const regionClass = showKSpaceBG ? SpaceToClass[space] : null;
return (
<>
{visible && (
@@ -147,7 +160,11 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
</div>
)}
<div className={clsx(classes.RootCustomNode, classes[STATUS_CLASSES[status]], { [classes.selected]: selected })}>
<div
className={clsx(classes.RootCustomNode, regionClass, classes[STATUS_CLASSES[status]], {
[classes.selected]: selected,
})}
>
{visible && (
<>
<div className={classes.HeadRow}>
@@ -183,7 +200,13 @@ export const SolarSystemNode = memo(({ data, selected }: WrapNodeProps<MapSolarS
)}
{!isWormhole && !customName && (
<div className="text-stone-400 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5">
<div
className={clsx('text-stone-400 whitespace-nowrap overflow-hidden text-ellipsis mr-0.5', {
['text-teal-100 font-bold']: space === Spaces.Caldari,
['text-yellow-100 font-bold']: space === Spaces.Amarr || space === Spaces.Matar,
['text-lime-200/80 font-bold']: space === Spaces.Gallente,
})}
>
{region_name}
</div>
)}

View File

@@ -11,3 +11,7 @@ export const isKnownSpace = (wormholeClassID: number) => {
return false;
};
export const isPossibleSpace = (spaces: number[], wormholeClassID: number) => {
return spaces.includes(wormholeClassID);
};

View File

@@ -54,7 +54,7 @@ export const RoutesWidgetContent = () => {
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [hubs, systems, systemStatics, lastUpdateKey]);
const preparedRoutes = useMemo(() => {
const preparedRoutes: Route[] = useMemo(() => {
return (
routes?.routes
.sort(sortByDist)
@@ -71,15 +71,17 @@ export const RoutesWidgetContent = () => {
);
}, [routes?.routes, routes?.systems_static_data, systemId]);
const refData = useRef({ open, loadSystems });
refData.current = { open, loadSystems };
const refData = useRef({ open, loadSystems, preparedRoutes });
refData.current = { open, loadSystems, preparedRoutes };
useEffect(() => {
(async () => await refData.current.loadSystems(hubs))();
}, [hubs]);
const handleClick = useCallback((e: MouseEvent, systemId: string) => {
refData.current.open(e, systemId);
const route = refData.current.preparedRoutes.find(x => x.destination.toString() === systemId);
refData.current.open(e, systemId, route?.mapped_systems ?? []);
}, []);
const handleContextMenu = useCallback(
@@ -146,7 +148,14 @@ export const RoutesWidgetContent = () => {
</div>
)}
<ContextMenuSystemInfo hubs={hubs} systems={systems} systemStatics={systemStatics} {...systemCtxProps} />
<ContextMenuSystemInfo
hubs={hubs}
routes={preparedRoutes}
systems={systems}
systemStatics={systemStatics}
systemIdFrom={systemId}
{...systemCtxProps}
/>
</>
);
};

View File

@@ -13,6 +13,8 @@ interface RightBarProps {
export const RightBar = ({ onShowOnTheMap }: RightBarProps) => {
const { outCommand, interfaceSettings, setInterfaceSettings } = useMapRootState();
const isShowMinimap = interfaceSettings.isShowMinimap === undefined ? true : interfaceSettings.isShowMinimap;
const handleAddCharacter = useCallback(() => {
outCommand({
type: OutCommand.addCharacter,
@@ -27,6 +29,13 @@ export const RightBar = ({ onShowOnTheMap }: RightBarProps) => {
}));
}, [setInterfaceSettings]);
const toggleKSpace = useCallback(() => {
setInterfaceSettings(x => ({
...x,
isShowKSpace: !x.isShowKSpace,
}));
}, [setInterfaceSettings]);
const toggleMenu = useCallback(() => {
setInterfaceSettings(x => ({
...x,
@@ -67,19 +76,31 @@ export const RightBar = ({ onShowOnTheMap }: RightBarProps) => {
<div className="flex flex-col items-center mb-2 gap-1">
<WdTooltipWrapper
content={interfaceSettings.isShowMinimap ? 'Hide minimap' : 'Show minimap'}
content={
interfaceSettings.isShowKSpace ? 'Hide highlighting Imperial Space' : 'Show highlighting Imperial Space'
}
position={TooltipPosition.left}
>
<button
className="btn bg-transparent text-gray-400 hover:text-white border-transparent hover:bg-transparent py-2 h-auto min-h-auto"
type="button"
onClick={toggleKSpace}
>
{interfaceSettings.isShowKSpace ? (
<i className="pi pi-star-fill text-lg"></i>
) : (
<i className="pi pi-star text-lg"></i>
)}
</button>
</WdTooltipWrapper>
<WdTooltipWrapper content={isShowMinimap ? 'Hide minimap' : 'Show minimap'} 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={toggleMinimap}
>
{interfaceSettings.isShowMinimap ? (
<i className="pi pi-eye text-lg"></i>
) : (
<i className="pi pi-eye-slash text-lg"></i>
)}
{isShowMinimap ? <i className="pi pi-eye text-lg"></i> : <i className="pi pi-eye-slash text-lg"></i>}
</button>
</WdTooltipWrapper>

View File

@@ -24,7 +24,7 @@ export const MapWrapper = ({ refn }: MapWrapperProps) => {
update,
outCommand,
data: { selectedConnections, selectedSystems, hubs, systems },
interfaceSettings: { isShowMenu, isShowMinimap = STORED_INTERFACE_DEFAULT_VALUES.isShowMinimap },
interfaceSettings: { isShowMenu, isShowMinimap = STORED_INTERFACE_DEFAULT_VALUES.isShowMinimap, isShowKSpace },
} = useMapRootState();
const { open, ...systemContextProps } = useContextMenuSystemHandlers({ systems, hubs, outCommand });
@@ -99,6 +99,7 @@ export const MapWrapper = ({ refn }: MapWrapperProps) => {
onSelectionContextMenu={handleSystemMultipleContext}
minimapClasses={!isShowMenu ? classes.MiniMap : undefined}
isShowMinimap={isShowMinimap}
showKSpaceBG={isShowKSpace}
/>
{openSettings != null && (

View File

@@ -5,3 +5,62 @@ export enum SESSION_KEY {
}
export const GRADIENT_MENU_ACTIVE_CLASSES = 'bg-gradient-to-br from-transparent/10 to-fuchsia-300/10';
export enum Regions {
Derelik = 10000001,
TheForge = 10000002,
Lonetrek = 10000016,
SinqLaison = 10000032,
Aridia = 10000054,
BlackRise = 10000069,
TheBleakLands = 10000038,
TheCitadel = 10000033,
Devoid = 10000036,
Domain = 10000043,
Essence = 10000064,
Everyshore = 10000037,
Genesis = 10000067,
Heimatar = 10000030,
Kador = 10000052,
Khanid = 10000049,
KorAzor = 10000065,
Metropolis = 10000042,
MoldenHeath = 10000028,
Placid = 10000048,
Solitude = 10000044,
TashMurkon = 10000020,
VergeVendor = 10000068,
}
export enum Spaces {
'Caldari' = 'Caldari',
'Gallente' = 'Gallente',
'Matar' = 'Matar',
'Amarr' = 'Amarr',
}
export const REGIONS_MAP: Record<number, Spaces> = {
[Regions.Derelik]: Spaces.Amarr,
[Regions.TheForge]: Spaces.Caldari,
[Regions.Lonetrek]: Spaces.Caldari,
[Regions.SinqLaison]: Spaces.Gallente,
[Regions.Aridia]: Spaces.Amarr,
[Regions.BlackRise]: Spaces.Caldari,
[Regions.TheBleakLands]: Spaces.Amarr,
[Regions.TheCitadel]: Spaces.Caldari,
[Regions.Devoid]: Spaces.Amarr,
[Regions.Domain]: Spaces.Amarr,
[Regions.Essence]: Spaces.Gallente,
[Regions.Everyshore]: Spaces.Gallente,
[Regions.Genesis]: Spaces.Amarr,
[Regions.Heimatar]: Spaces.Matar,
[Regions.Kador]: Spaces.Amarr,
[Regions.Khanid]: Spaces.Amarr,
[Regions.KorAzor]: Spaces.Amarr,
[Regions.Metropolis]: Spaces.Matar,
[Regions.MoldenHeath]: Spaces.Matar,
[Regions.Placid]: Spaces.Gallente,
[Regions.Solitude]: Spaces.Gallente,
[Regions.TashMurkon]: Spaces.Amarr,
[Regions.VergeVendor]: Spaces.Gallente,
};

View File

@@ -23,7 +23,7 @@ export default {
onError: handleError,
});
this.pushEvent('loaded');
this.pushEvent('ui_loaded');
},
handleEventWrapper(event: string, handler: (payload: any) => void) {

View File

@@ -30,11 +30,13 @@ const INITIAL_DATA: MapRootData = {
type InterfaceStoredSettings = {
isShowMenu: boolean;
isShowMinimap: boolean;
isShowKSpace: boolean;
};
export const STORED_INTERFACE_DEFAULT_VALUES: InterfaceStoredSettings = {
isShowMenu: false,
isShowMinimap: true,
isShowKSpace: false,
};
export interface MapRootContextProps {
@@ -50,6 +52,7 @@ const MapRootContext = createContext<MapRootContextProps>({
update: () => {},
data: { ...INITIAL_DATA },
mapRef: { current: null },
// @ts-ignore
outCommand: async () => void 0,
interfaceSettings: STORED_INTERFACE_DEFAULT_VALUES,
setInterfaceSettings: () => null,

Binary file not shown.

After

Width:  |  Height:  |  Size: 403 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 340 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 509 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 603 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 162 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 95 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 188 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 224 KiB

View File

@@ -78,6 +78,8 @@ config :wanderer_app,
git_sha: System.get_env("GIT_SHA", "111"),
custom_route_base_url: System.get_env("CUSTOM_ROUTE_BASE_URL"),
invites: System.get_env("WANDERER_INVITES", "false") == "true",
admin_username: System.get_env("WANDERER_ADMIN_USERNAME", "admin"),
admin_password: System.get_env("WANDERER_ADMIN_PASSWORD"),
admins: admins,
corp_id: System.get_env("WANDERER_CORP_ID", "-1") |> String.to_integer(),
corp_wallet: System.get_env("WANDERER_CORP_WALLET", ""),

View File

@@ -43,7 +43,6 @@ defmodule WandererApp.Api.AccessList do
primary?(true)
argument :owner_id, :uuid, allow_nil?: false
argument :owner_id_text_input, :string, allow_nil?: true
change manage_relationship(:owner_id, :owner, on_lookup: :relate, on_no_match: nil)
end
@@ -51,8 +50,6 @@ defmodule WandererApp.Api.AccessList do
update :update do
accept [:name, :description, :owner_id]
primary?(true)
argument :owner_id_text_input, :string, allow_nil?: true
end
update :assign_owner do

View File

@@ -63,7 +63,6 @@ defmodule WandererApp.Api.Map do
primary?(true)
argument :owner_id, :uuid, allow_nil?: false
argument :owner_id_text_input, :string, allow_nil?: true
argument :create_default_acl, :boolean, allow_nil?: true
argument :acls, {:array, :uuid}, allow_nil?: true
argument :acls_text_input, :string, allow_nil?: true

View File

@@ -18,10 +18,7 @@ defmodule WandererApp.Api.MapConnection do
action: :read
)
define(:by_locations,
get_by: [:map_id, :solar_system_source, :solar_system_target],
action: :read
)
define(:by_locations, action: :read_by_locations)
define(:read_by_map, action: :read_by_map)
define(:get_link_pairs_advanced, action: :get_link_pairs_advanced)
@@ -47,6 +44,13 @@ defmodule WandererApp.Api.MapConnection do
filter(expr(map_id == ^arg(:map_id)))
end
read :read_by_locations do
argument(:map_id, :string, allow_nil?: false)
argument(:solar_system_source, :integer, allow_nil?: false)
argument(:solar_system_target, :integer, allow_nil?: false)
filter(expr(map_id == ^arg(:map_id) and solar_system_source == ^arg(:solar_system_source) and solar_system_target == ^arg(:solar_system_target)))
end
read :get_link_pairs_advanced do
argument(:map_id, :string, allow_nil?: false)
argument(:include_mass_crit, :boolean, allow_nil?: false)

View File

@@ -68,13 +68,16 @@ defmodule WandererApp.Character.TrackerManager.Impl do
state
false ->
WandererApp.Character.update_character_state(character_id, %{opts: opts})
:telemetry.execute([:wanderer_app, :character, :tracker, :started], %{count: 1})
Logger.debug(fn -> "Start character tracker: #{inspect(character_id)}" end)
tracked_characters = [character_id | state.characters] |> Enum.uniq()
Task.start_link(fn ->
WandererApp.Character.update_character_state(character_id, %{opts: opts})
:telemetry.execute([:wanderer_app, :character, :tracker, :started], %{count: 1})
:ok
end)
tracked_characters = [character_id | state.characters] |> Enum.uniq()
WandererApp.Cache.insert("tracked_characters", tracked_characters)
%{state | characters: tracked_characters}

View File

@@ -11,6 +11,8 @@ defmodule WandererApp.Env do
def map_subscriptions_enabled?, do: get_key(:map_subscriptions_enabled, false)
def wallet_tracking_enabled?, do: get_key(:wallet_tracking_enabled, false)
def admins, do: get_key(:admins, [])
def admin_username, do: get_key(:admin_username)
def admin_password, do: get_key(:admin_password)
def corp_wallet, do: get_key(:corp_wallet, "")
def corp_eve_id, do: get_key(:corp_id, -1)
def subscription_settings, do: get_key(:subscription_settings)

View File

@@ -229,7 +229,7 @@ defmodule WandererApp.EveDataService do
constellation_id = row["constellationID"] |> Integer.parse() |> elem(0)
{:ok, wormhole_class_id} =
_get_wormhole_class_id(
get_wormhole_class_id(
map_location_wormhole_classes,
region_id,
constellation_id,
@@ -237,16 +237,16 @@ defmodule WandererApp.EveDataService do
)
{:ok, constellation_name} =
_get_constellation_name(map_constellations, constellation_id)
get_constellation_name(map_constellations, constellation_id)
{:ok, region_name} = _get_region_name(map_regions, region_id)
{:ok, region_name} = get_region_name(map_regions, region_id)
{:ok, wormhole_class} = _get_wormhole_class(wormhole_classes, wormhole_class_id)
{:ok, wormhole_class} = get_wormhole_class(wormhole_classes, wormhole_class_id)
{:ok, security} = _get_security(row["security"])
{:ok, security} = get_security(row["security"])
{:ok, class_title} =
_get_class_title(
get_class_title(
wormhole_classes_info,
wormhole_class_id,
security,
@@ -270,7 +270,7 @@ defmodule WandererApp.EveDataService do
solar_system_id: solar_system_id,
solar_system_name: row["solarSystemName"],
solar_system_name_lc: row["solarSystemName"] |> String.downcase(),
sun_type_id: _get_sun_type_id(row["sunTypeID"]),
sun_type_id: get_sun_type_id(row["sunTypeID"]),
constellation_name: constellation_name,
region_name: region_name,
security: security,
@@ -279,8 +279,8 @@ defmodule WandererApp.EveDataService do
type_description: wormhole_class.title,
is_shattered: is_shattered
}
|> _get_wormhole_data(wormhole_systems, solar_system_id, wormhole_class)
|> _get_triglavian_data(triglavian_systems, solar_system_id)
|> get_wormhole_data(wormhole_systems, solar_system_id, wormhole_class)
|> get_triglavian_data(triglavian_systems, solar_system_id)
end
)
end
@@ -332,14 +332,14 @@ defmodule WandererApp.EveDataService do
)
end
defp _get_sun_type_id(sun_type_id) do
defp get_sun_type_id(sun_type_id) do
case sun_type_id do
"None" -> 0
_ -> sun_type_id |> Integer.parse() |> elem(0)
end
end
defp _get_wormhole_data(default_data, wormhole_systems, solar_system_id, wormhole_class) do
defp get_wormhole_data(default_data, wormhole_systems, solar_system_id, wormhole_class) do
case Enum.find(wormhole_systems, fn system -> system.solar_system_id == solar_system_id end) do
nil ->
default_data
@@ -355,7 +355,7 @@ defmodule WandererApp.EveDataService do
end
end
defp _get_triglavian_data(default_data, triglavian_systems, solar_system_id) do
defp get_triglavian_data(default_data, triglavian_systems, solar_system_id) do
case Enum.find(triglavian_systems, fn system -> system.solar_system_id == solar_system_id end) do
nil ->
default_data
@@ -370,14 +370,18 @@ defmodule WandererApp.EveDataService do
end
end
defp _get_security(security) do
defp get_security(security) do
case security do
nil -> {:ok, ""}
_ -> {:ok, Decimal.parse(security) |> elem(0) |> Decimal.round(1) |> Decimal.to_string()}
_ -> {:ok, String.to_float(security) |> get_true_security() |> Float.to_string(decimals: 1)}
end
end
defp _get_class_title(wormhole_classes_info, wormhole_class_id, security, wormhole_class) do
defp get_true_security(security) when is_float(security) and security > 0.0 and security < 0.05, do: security |> Float.ceil(1)
defp get_true_security(security) when is_float(security), do: security |> Float.floor(1)
defp get_class_title(wormhole_classes_info, wormhole_class_id, security, wormhole_class) do
case wormhole_class_id in [
wormhole_classes_info.names["hs"],
wormhole_classes_info.names["ls"],
@@ -391,7 +395,7 @@ defmodule WandererApp.EveDataService do
end
end
defp _get_constellation_name(constellations, constellation_id) do
defp get_constellation_name(constellations, constellation_id) do
case Enum.find(constellations, fn constellation ->
constellation.constellation_id == constellation_id
end) do
@@ -400,24 +404,24 @@ defmodule WandererApp.EveDataService do
end
end
defp _get_region_name(regions, region_id) do
defp get_region_name(regions, region_id) do
case Enum.find(regions, fn region -> region.region_id == region_id end) do
nil -> {:ok, ""}
region -> {:ok, region.region_name}
end
end
defp _get_wormhole_class(wormhole_classes, wormhole_class_id) do
defp get_wormhole_class(wormhole_classes, wormhole_class_id) do
{:ok,
Enum.find(wormhole_classes, fn wormhole_class ->
wormhole_class.wormhole_class_id == wormhole_class_id
end)}
end
defp _get_wormhole_class_id(_systems, _region_id, _constellation_id, 30_100_000),
defp get_wormhole_class_id(_systems, _region_id, _constellation_id, 30_100_000),
do: {:ok, 10_100}
defp _get_wormhole_class_id(systems, region_id, constellation_id, solar_system_id) do
defp get_wormhole_class_id(systems, region_id, constellation_id, solar_system_id) do
with region <-
Enum.find(systems, fn system ->
system.location_id |> Integer.parse() |> elem(0) == region_id
@@ -430,23 +434,23 @@ defmodule WandererApp.EveDataService do
Enum.find(systems, fn system ->
system.location_id |> Integer.parse() |> elem(0) == solar_system_id
end),
wormhole_class_id <- _get_wormhole_class_id(region, constellation, solar_system) do
wormhole_class_id <- get_wormhole_class_id(region, constellation, solar_system) do
{:ok, wormhole_class_id}
else
_ -> {:ok, -1}
end
end
defp _get_wormhole_class_id(_region, _constellation, solar_system)
defp get_wormhole_class_id(_region, _constellation, solar_system)
when not is_nil(solar_system),
do: solar_system.wormhole_class_id |> Integer.parse() |> elem(0)
defp _get_wormhole_class_id(_region, constellation, _solar_system)
defp get_wormhole_class_id(_region, constellation, _solar_system)
when not is_nil(constellation),
do: constellation.wormhole_class_id |> Integer.parse() |> elem(0)
defp _get_wormhole_class_id(region, _constellation, _solar_system) when not is_nil(region),
defp get_wormhole_class_id(region, _constellation, _solar_system) when not is_nil(region),
do: region.wormhole_class_id |> Integer.parse() |> elem(0)
defp _get_wormhole_class_id(_region, _constellation, _solar_system), do: -1
defp get_wormhole_class_id(_region, _constellation, _solar_system), do: -1
end

View File

@@ -10,8 +10,8 @@ defmodule WandererApp.Map.Manager do
alias WandererApp.Map.Server
alias WandererApp.Map.ServerSupervisor
@maps_start_per_second 100
@maps_start_interval 1500
@maps_start_per_second 5
@maps_start_interval 1000
@maps_queue :maps_queue
@garbage_collection_interval :timer.hours(1)
@check_maps_queue_interval :timer.seconds(1)

View File

@@ -75,11 +75,11 @@ defmodule WandererApp.Map.Server do
|> map_pid!
|> GenServer.call({&Impl.get_characters/1, []}, :timer.minutes(1))
def add_character(map_id, character) when is_binary(map_id),
def add_character(map_id, character, track_character \\ false) when is_binary(map_id),
do:
map_id
|> map_pid!
|> GenServer.cast({&Impl.add_character/2, [character]})
|> GenServer.cast({&Impl.add_character/3, [character, track_character]})
def remove_character(map_id, character_id) when is_binary(map_id),
do:

View File

@@ -176,38 +176,46 @@ defmodule WandererApp.Map.Server.Impl do
def get_characters(%{map_id: map_id} = _state),
do: {:ok, map_id |> WandererApp.Map.list_characters()}
def add_character(%{map_id: map_id} = state, %{id: character_id} = character) do
with :ok <- map_id |> WandererApp.Map.add_character(character),
{:ok, _} <-
WandererApp.MapCharacterSettingsRepo.create(%{
character_id: character_id,
map_id: map_id,
tracked: false
}),
{:ok, character} <- WandererApp.Character.get_character(character_id) do
broadcast!(map_id, :character_added, character)
def add_character(%{map_id: map_id} = state, %{id: character_id} = character, track_character) do
Task.start_link(fn ->
with :ok <- map_id |> WandererApp.Map.add_character(character),
{:ok, _} <-
WandererApp.MapCharacterSettingsRepo.create(%{
character_id: character_id,
map_id: map_id,
tracked: track_character
}),
{:ok, character} <- WandererApp.Character.get_character(character_id) do
broadcast!(map_id, :character_added, character)
:telemetry.execute([:wanderer_app, :map, :character, :added], %{count: 1})
:telemetry.execute([:wanderer_app, :map, :character, :added], %{count: 1})
state
else
{:error, _error} ->
state
end
:ok
else
{:error, _error} ->
:ok
end
end)
state
end
def remove_character(%{map_id: map_id} = state, character_id) do
with :ok <- WandererApp.Map.remove_character(map_id, character_id),
{:ok, character} <- WandererApp.Character.get_character(character_id) do
broadcast!(map_id, :character_removed, character)
Task.start_link(fn ->
with :ok <- WandererApp.Map.remove_character(map_id, character_id),
{:ok, character} <- WandererApp.Character.get_character(character_id) do
broadcast!(map_id, :character_removed, character)
:telemetry.execute([:wanderer_app, :map, :character, :removed], %{count: 1})
:telemetry.execute([:wanderer_app, :map, :character, :removed], %{count: 1})
state
else
{:error, _error} ->
state
end
:ok
else
{:error, _error} ->
:ok
end
end)
state
end
def untrack_characters(%{map_id: map_id} = state, characters_ids) do
@@ -358,14 +366,18 @@ defmodule WandererApp.Map.Server.Impl do
|> Enum.each(fn connection ->
@logger.debug(fn -> "Removing connection from map: #{inspect(connection)}" end)
connection
|> WandererApp.MapConnectionRepo.destroy!()
{:ok, from_connections} = WandererApp.MapConnectionRepo.get_by_locations(map_id, connection.solar_system_source, connection.solar_system_target)
{:ok, to_connections} = WandererApp.MapConnectionRepo.get_by_locations(map_id, connection.solar_system_target, connection.solar_system_source)
[from_connections ++ to_connections]
|> List.flatten()
|> WandererApp.MapConnectionRepo.bulk_destroy!()
|> case do
:ok ->
:ok
{:error, error} ->
@logger.error("Failed to remove connection from map: #{inspect(error, pretty: true)}")
error ->
@logger.error("Failed to remove connections from map: #{inspect(error, pretty: true)}")
:ok
end
end)

View File

@@ -1,12 +1,43 @@
defmodule WandererApp.MapConnectionRepo do
use WandererApp, :repository
require Logger
@logger Application.compile_env(:wanderer_app, :logger)
def get_by_map(map_id),
do: WandererApp.Api.MapConnection.read_by_map(%{map_id: map_id})
def get_by_locations(map_id, solar_system_source, solar_system_target) do
WandererApp.Api.MapConnection.by_locations(%{map_id: map_id, solar_system_source: solar_system_source, solar_system_target: solar_system_target})
|> case do
{:ok, connections} ->
{:ok, connections}
{:error, %Ash.Error.Query.NotFound{}} ->
{:ok, []}
{:error, error} ->
@logger.error("Failed to get connections: #{inspect(error, pretty: true)}")
{:error, error}
end
end
def create!(connection), do: connection |> WandererApp.Api.MapConnection.create!()
def destroy!(connection), do: connection |> WandererApp.Api.MapConnection.destroy!()
def destroy!(connection), do:
connection |> WandererApp.Api.MapConnection.destroy!()
def bulk_destroy!(connections) do
connections
|> WandererApp.Api.MapConnection.destroy!()
|> case do
%Ash.BulkResult{status: :success} ->
:ok
error ->
error
end
end
def update_time_status(connection, update),
do:

View File

@@ -602,11 +602,11 @@ defmodule WandererAppWeb.CoreComponents do
<tr
:for={row <- @rows}
id={@row_id && @row_id.(row)}
phx-click={@row_click && @row_click.(row)}
class={"hover #{if @row_selected && @row_selected.(row), do: "!bg-slate-600", else: ""} #{if @row_click, do: "cursor-pointer", else: ""}"}
>
<td
:for={{col, _index} <- Enum.with_index(@col)}
phx-click={@row_click && @row_click.(row)}
>
<%= render_slot(col, @row_item.(row)) %>
</td>

View File

@@ -0,0 +1,13 @@
defmodule WandererAppWeb.BasicAuth do
@moduledoc false
def admin_basic_auth(conn, _opts) do
admin_password = WandererApp.Env.admin_password()
if is_nil(admin_password) do
conn
else
conn
|> Plug.BasicAuth.basic_auth(username: WandererApp.Env.admin_username(), password: admin_password)
end
end
end

View File

@@ -99,40 +99,34 @@ defmodule WandererAppWeb.AccessListsLive do
end
defp apply_action(socket, :add_members, %{"id" => acl_id} = _params) do
{:ok, %{owner: %{id: _character_id}} = access_list} =
socket.assigns.access_lists |> Enum.find(&(&1.id == acl_id)) |> Ash.load(:owner)
with {:ok, %{owner: %{id: _character_id}} = access_list} <- socket.assigns.access_lists |> Enum.find(&(&1.id == acl_id)) |> Ash.load(:owner),
user_character_ids <- socket.assigns.current_user.characters |> Enum.map(& &1.id) do
user_character_ids
|> Enum.each(fn user_character_id ->
:ok = WandererApp.Character.TrackerManager.start_tracking(user_character_id)
end)
user_character_ids = socket.assigns.current_user.characters |> Enum.map(& &1.id)
user_character_ids
|> Enum.each(fn user_character_id ->
:ok = WandererApp.Character.TrackerManager.start_tracking(user_character_id)
end)
socket
|> assign(:active_page, :access_lists)
|> assign(:page_title, "Access Lists - Add Members")
|> assign(:selected_acl_id, acl_id)
|> assign(:user_character_ids, user_character_ids)
|> assign(
member_search_options: socket.assigns.characters |> Enum.map(&map_user_character_info/1)
)
|> assign(:access_list, access_list)
|> assign(
:members,
WandererApp.Api.AccessListMember.read_by_access_list!(%{access_list_id: acl_id})
)
|> assign(
:member_form,
%{} |> to_form()
)
end
@impl true
def handle_event("set-default", %{"id" => id}, socket) do
send_update(LiveSelect.Component, options: socket.assigns.characters, id: id)
{:noreply, socket}
socket
|> assign(:active_page, :access_lists)
|> assign(:page_title, "Access Lists - Add Members")
|> assign(:selected_acl_id, acl_id)
|> assign(:user_character_ids, user_character_ids)
|> assign(
member_search_options: socket.assigns.characters |> Enum.map(&map_user_character_info/1)
)
|> assign(:access_list, access_list)
|> assign(
:members,
WandererApp.Api.AccessListMember.read_by_access_list!(%{access_list_id: acl_id})
)
|> assign(
:member_form,
%{} |> to_form()
)
else
_ ->
socket
end
end
@impl true

View File

@@ -127,23 +127,13 @@
<.form :let={f} for={@form} phx-change="validate" phx-submit={@live_action}>
<.input type="text" field={f[:name]} placeholder="Name" />
<.input type="textarea" field={f[:description]} placeholder="Public description" />
<.live_select
<.input
type="select"
field={f[:owner_id]}
dropdown_extra_class="max-h-64"
available_option_class="w-full"
value_mapper={&map_character/1}
update_min_len={0}
phx-focus="set-default"
options={@characters}
placeholder="Owner"
>
<:option :let={option}>
<div class="flex items-center">
<.avatar url={member_icon_url(option.eve_id)} label={option.label} />
&nbsp;<%= option.label %>
</div>
</:option>
</.live_select>
class="p-dropdown p-component p-inputwrapper mt-8"
placeholder="Select a map owner"
options={Enum.map(@characters, fn character -> {character.label, character.id} end)}
/>
<div class="modal-action">
<.button class="mt-2" type="submit" phx-disable-with="Saving...">
<%= (@live_action == :create && "Create") || "Save" %>

View File

@@ -42,8 +42,10 @@ defmodule WandererAppWeb.CharactersTrackingLive do
{:ok, character_settings} =
case WandererApp.Api.MapCharacterSettings.read_by_map(%{map_id: selected_map.id}) do
{:ok, settings} -> {:ok, settings}
_ -> {:ok, []}
{:ok, settings} ->
{:ok, settings}
_ ->
{:ok, []}
end
user_id = socket.assigns.user_id

View File

@@ -8,7 +8,7 @@ defmodule WandererAppWeb.MapLive do
socket =
with %{"slug" => map_slug} <- params do
socket
|> _init_state(map_slug, false)
|> _init_state(map_slug)
else
_ ->
# redirect back to main
@@ -29,7 +29,7 @@ defmodule WandererAppWeb.MapLive do
{:ok, socket |> assign(server_online: false)}
end
defp _init_state(socket, map_slug, is_reconnect?) do
defp _init_state(socket, map_slug) do
current_user = socket.assigns.current_user
ErrorTracker.set_context(%{user_id: current_user.id})
@@ -44,7 +44,8 @@ defmodule WandererAppWeb.MapLive do
id: map_id,
deleted: false
} = map} ->
Task.async(fn -> _load_initial_data(map, is_reconnect?, current_user) end)
Process.send_after(self(), {:init_map, map}, 10)
socket
|> assign(
@@ -113,7 +114,7 @@ defmodule WandererAppWeb.MapLive do
}
} = socket
) do
Task.async(fn -> _on_map_started(map_id, current_user, user_permissions) end)
_on_map_started(map_id, current_user, user_permissions)
{:noreply, socket}
end
@@ -330,7 +331,7 @@ defmodule WandererAppWeb.MapLive do
_ ->
:ok = _track_characters(map_characters, map_id, true)
:ok = _add_characters(map_characters, map_id)
:ok = _add_characters(map_characters, map_id, track_character)
end
socket
@@ -344,6 +345,168 @@ defmodule WandererAppWeb.MapLive do
{:noreply, socket}
end
def handle_info({:init_map, map}, %{assigns: %{current_user: current_user}} = socket) do
with %{
id: map_id,
deleted: false,
only_tracked_characters: only_tracked_characters,
user_permissions: user_permissions,
name: map_name,
owner_id: owner_id
} <- map do
user_permissions =
WandererApp.Permissions.get_map_permissions(
user_permissions,
owner_id,
current_user.characters |> Enum.map(& &1.id)
)
{:ok, character_settings} =
case WandererApp.Api.MapCharacterSettings.read_by_map(%{map_id: map_id}) do
{:ok, settings} -> {:ok, settings}
_ -> {:ok, []}
end
{:ok, %{characters: availaible_map_characters}} =
WandererApp.Maps.load_characters(map, character_settings, current_user.id)
can_view? = user_permissions.view_system
can_track? = user_permissions.track_character
tracked_character_ids =
availaible_map_characters |> Enum.filter(& &1.tracked) |> Enum.map(& &1.id)
all_character_tracked? = (not (availaible_map_characters |> Enum.empty?())) and availaible_map_characters |> Enum.all?(& &1.tracked)
cond do
(only_tracked_characters and can_track? and all_character_tracked?) or
(not only_tracked_characters and can_view?) ->
Process.send_after(self(), {:map_init, %{
map_id: map_id,
page_title: map_name,
user_permissions: user_permissions,
tracked_character_ids: tracked_character_ids
}}, 10)
only_tracked_characters and can_track? and not all_character_tracked? ->
Process.send_after(self(), :not_all_characters_tracked, 10)
true ->
Process.send_after(self(), :no_permissions, 10)
end
else
_ ->
Process.send_after(self(), :no_access, 10)
end
{:noreply, socket}
end
def handle_info({:map_init, %{map_id: map_id} = initial_data}, socket) do
Phoenix.PubSub.subscribe(WandererApp.PubSub, map_id)
{:noreply,
socket
|> assign(initial_data)}
end
def handle_info({:map_start,
%{
map_id: map_id,
user_characters: user_character_eve_ids,
initial_data: initial_data,
events: events
} = _started_data}, socket) do
socket =
events
|> Enum.reduce(socket, fn event, socket ->
case event do
{:track_characters, map_characters, track_character} ->
:ok = _track_characters(map_characters, map_id, track_character)
:ok = _add_characters(map_characters, map_id, track_character)
socket
:invalid_token_message ->
socket
|> put_flash(
:error,
"One of your characters has expired token. Please refresh it on characters page."
)
:empty_tracked_characters ->
socket
|> put_flash(
:info,
"You should enable tracking for at least one character to work with map."
)
:map_character_limit ->
socket
|> put_flash(
:error,
"Map reached its character limit, your characters won't be tracked. Please contact administrator."
)
_ ->
socket
end
end)
Process.send_after(self(), {:map_loaded,
%{
map_id: map_id,
user_characters: user_character_eve_ids,
initial_data: initial_data
}}, 10)
{:noreply, socket}
end
def handle_info({:map_loaded,
%{
map_id: map_id,
user_characters: user_character_eve_ids,
initial_data: initial_data
} = _loaded_data}, socket) do
map_characters = map_id |> WandererApp.Map.list_characters()
{:noreply,
socket
|> assign(
map_loaded?: true,
user_characters: user_character_eve_ids,
has_tracked_characters?: _has_tracked_characters?(user_character_eve_ids)
)
|> _push_map_event("init", initial_data |> Map.merge(%{
characters: map_characters |> Enum.map(&map_ui_character/1)
}))
|> push_event("js-exec", %{
to: "#map-loader",
attr: "data-loaded"
})}
end
def handle_info(:no_access, socket), do:
{:noreply,
socket
|> put_flash(:error, "You don't have an access to this map.")
|> push_navigate(to: ~p"/maps")}
def handle_info(:no_permissions, socket), do:
{:noreply,
socket
|> put_flash(:error, "You don't have permissions to use this map.")
|> push_navigate(to: ~p"/maps")}
def handle_info(:not_all_characters_tracked, socket), do:
{:noreply,
socket
|> put_flash(
:error,
"You should enable tracking for all characters that have access to this map first!"
)
|> push_navigate(to: ~p"/tracking/#{socket.assigns.map_slug}")}
@impl true
def handle_info(
{ref, result},
@@ -360,74 +523,13 @@ defmodule WandererAppWeb.MapLive do
maps: maps
)}
{:map_init_data, %{map_id: map_id} = initial_data} ->
Phoenix.PubSub.subscribe(WandererApp.PubSub, map_id)
WandererApp.Map.Manager.start_map(map_id)
{:map_started_data, started_data} ->
Process.send_after(self(), {:map_start, started_data}, 100)
{:noreply, socket}
{:ok, map_started} = WandererApp.Cache.lookup("map_#{map_id}:started", false)
if map_started do
Process.send_after(self(), %{event: :map_started}, 100)
end
{:noreply,
socket
|> assign(initial_data)}
{:map_started,
%{
map_id: map_id,
user_characters: user_character_eve_ids,
initial_data: initial_data,
events: events
} = _started_data} ->
socket =
events
|> Enum.reduce(socket, fn event, socket ->
case event do
{:track_characters, map_characters, track_character} ->
:ok = _track_characters(map_characters, map_id, track_character)
:ok = _add_characters(map_characters, map_id)
socket
:invalid_token_message ->
socket
|> put_flash(
:error,
"One of your characters has expired token. Please refresh it on characters page."
)
:empty_tracked_characters ->
socket
|> put_flash(
:info,
"You should enable tracking for at least one character to work with map."
)
:map_character_limit ->
socket
|> put_flash(
:error,
"Map reached its character limit, your characters won't be tracked. Please contact administrator."
)
_ ->
socket
end
end)
{:noreply,
socket
|> assign(
map_loaded?: true,
user_characters: user_character_eve_ids,
has_tracked_characters?: _has_tracked_characters?(user_character_eve_ids)
)
|> _push_map_event("init", initial_data)
|> push_event("js-exec", %{
to: "#map-loader",
attr: "data-loaded"
})}
{:map_error, map_error} ->
Process.send_after(self(), map_error, 100)
{:noreply, socket}
{:character_activity, character_activity} ->
{:noreply,
@@ -447,37 +549,26 @@ defmodule WandererAppWeb.MapLive do
}
)}
{:map_error, :not_all_characters_tracked} ->
{:noreply,
socket
|> put_flash(
:error,
"You should enable tracking for all characters that have access to this map first!"
)
|> push_navigate(to: ~p"/tracking/#{socket.assigns.map_slug}")}
{:map_error, :no_permissions} ->
{:noreply,
socket
|> put_flash(:error, "You don't have permissions to use this map.")
|> push_navigate(to: ~p"/maps")}
{:map_error, :no_access} ->
{:noreply,
socket
|> put_flash(:error, "You don't have an access to this map.")
|> push_navigate(to: ~p"/maps")}
_ ->
{:noreply, socket}
end
end
@impl true
def handle_info(_event, socket), do: {:noreply, socket}
@impl true
def handle_event("loaded", _body, socket) do
def handle_event("ui_loaded", _body, %{assigns: %{map_id: map_id}} = socket) do
{:ok, map_started} = WandererApp.Cache.lookup("map_#{map_id}:started", false)
if map_started do
Process.send_after(self(), %{event: :map_started}, 10)
else
WandererApp.Map.Manager.start_map(map_id)
end
{:noreply, socket}
end
@@ -487,7 +578,8 @@ defmodule WandererAppWeb.MapLive do
end
@impl true
def handle_event("change_map", %{"map_slug" => map_slug} = _event, socket) do
def handle_event("change_map", %{"map_slug" => map_slug} = _event, %{assigns: %{map_id: map_id}} = socket) do
Phoenix.PubSub.unsubscribe(WandererApp.PubSub, map_id)
{:noreply, push_navigate(socket, to: ~p"/#{map_slug}")}
end
@@ -1250,7 +1342,6 @@ defmodule WandererAppWeb.MapLive do
def handle_event("toggle_track", %{"character-id" => character_id}, socket) do
map = socket.assigns.map
character_settings = socket.assigns.character_settings
current_user = socket.assigns.current_user
socket =
case character_settings |> Enum.find(&(&1.character_id == character_id)) do
@@ -1265,7 +1356,7 @@ defmodule WandererAppWeb.MapLive do
character = map_character_settings |> Ash.load!(:character) |> Map.get(:character)
:ok = _track_characters([character], map.id, true)
:ok = _add_characters([character], map.id)
:ok = _add_characters([character], map.id, true)
socket
@@ -1300,7 +1391,7 @@ defmodule WandererAppWeb.MapLive do
character = map_character_settings |> Ash.load!(:character) |> Map.get(:character)
:ok = _track_characters([character], map.id, true)
:ok = _add_characters([character], map.id)
:ok = _add_characters([character], map.id, true)
socket
end
@@ -1308,7 +1399,7 @@ defmodule WandererAppWeb.MapLive do
%{result: characters} = socket.assigns.characters
{:ok, map_characters} = _get_map_characters(map.id, socket.assigns.current_user)
{:ok, map_characters} = _get_tracked_map_characters(map.id, socket.assigns.current_user)
user_character_eve_ids = map_characters |> Enum.map(& &1.eve_id)
@@ -1392,74 +1483,17 @@ defmodule WandererAppWeb.MapLive do
{:noreply, socket}
end
defp _load_initial_data(map, is_reconnect?, current_user) do
with %{
id: map_id,
deleted: false,
only_tracked_characters: only_tracked_characters,
user_permissions: user_permissions,
name: map_name,
owner_id: owner_id
} <- map do
user_permissions =
WandererApp.Permissions.get_map_permissions(
user_permissions,
owner_id,
current_user.characters |> Enum.map(& &1.id)
)
{:ok, character_settings} =
case WandererApp.Api.MapCharacterSettings.read_by_map(%{map_id: map_id}) do
{:ok, settings} -> {:ok, settings}
_ -> {:ok, []}
end
{:ok, %{characters: availaible_map_characters}} =
WandererApp.Maps.load_characters(map, character_settings, current_user.id)
can_view? = user_permissions.view_system
can_track? = user_permissions.track_character
tracked_character_ids =
availaible_map_characters |> Enum.filter(& &1.tracked) |> Enum.map(& &1.id)
all_character_tracked? = availaible_map_characters |> Enum.all?(& &1.tracked)
cond do
(only_tracked_characters and can_track? and all_character_tracked?) or
(not only_tracked_characters and can_view?) ->
{:map_init_data,
%{
map_id: map_id,
page_title: map_name,
user_permissions: user_permissions,
tracked_character_ids: tracked_character_ids,
is_reconnect?: is_reconnect?
}}
only_tracked_characters and can_track? and not all_character_tracked? ->
{:map_error, :not_all_characters_tracked}
true ->
{:map_error, :no_permissions}
end
else
_ ->
{:map_error, :no_access}
end
end
defp _on_map_started(map_id, current_user, user_permissions) do
case user_permissions do
%{view_system: true, track_character: track_character} ->
{:ok, _} = current_user |> WandererApp.Api.User.update_last_map(%{last_map_id: map_id})
{:ok, map_characters} = _get_map_characters(map_id, current_user)
{:ok, tracked_map_characters} = _get_tracked_map_characters(map_id, current_user)
user_character_eve_ids = map_characters |> Enum.map(& &1.eve_id)
user_character_eve_ids = tracked_map_characters |> Enum.map(& &1.eve_id)
events =
case map_characters |> Enum.any?(&(&1.access_token == nil)) do
case tracked_map_characters |> Enum.any?(&(&1.access_token == nil)) do
true ->
[:invalid_token_message]
@@ -1468,7 +1502,7 @@ defmodule WandererAppWeb.MapLive do
end
events =
case map_characters |> Enum.empty?() do
case tracked_map_characters |> Enum.empty?() do
true ->
events ++ [:empty_tracked_characters]
@@ -1484,21 +1518,18 @@ defmodule WandererAppWeb.MapLive do
events =
case present_character_ids |> Enum.count() < characters_limit do
true ->
events ++ [{:track_characters, map_characters, track_character}]
events ++ [{:track_characters, tracked_map_characters, track_character}]
_ ->
events ++ [:map_character_limit]
end
characters = map_id |> WandererApp.Map.list_characters() |> Enum.map(&map_ui_character/1)
{:ok, kills} = WandererApp.Cache.lookup("map_#{map_id}:zkb_kills", Map.new())
initial_data =
map_id
|> _get_map_data()
|> Map.merge(%{
characters: characters,
kills:
kills
|> Enum.filter(fn {_, kills} -> kills > 0 end)
@@ -1535,16 +1566,23 @@ defmodule WandererAppWeb.MapLive do
)
|> Map.put(:reset, true)
{:map_started,
%{
map_id: map_id,
user_characters: user_character_eve_ids,
initial_data: initial_data,
events: events
}}
Process.send_after(self(), {:map_start, %{
map_id: map_id,
user_characters: user_character_eve_ids,
initial_data: initial_data,
events: events
}}, 10)
# {:map_started_data,
# %{
# map_id: map_id,
# user_characters: user_character_eve_ids,
# initial_data: initial_data,
# events: events
# }}
_ ->
{:map_error, :no_access}
Process.send_after(self(), {:map_error, :no_access}, 10)
end
end
@@ -1590,7 +1628,7 @@ defmodule WandererAppWeb.MapLive do
}
end
defp _get_map_characters(map_id, current_user) do
defp _get_tracked_map_characters(map_id, current_user) do
case WandererApp.Api.MapCharacterSettings.tracked_by_map(%{
map_id: map_id,
character_ids: current_user.characters |> Enum.map(& &1.id)
@@ -1923,17 +1961,13 @@ defmodule WandererAppWeb.MapLive do
defp _get_routes_settings(_), do: %{}
defp _add_characters([], _map_id), do: :ok
defp _add_characters([], _map_id, _track_character), do: :ok
defp _add_characters([character | characters], map_id) do
Task.async(fn ->
map_id
|> WandererApp.Map.Server.add_character(character)
defp _add_characters([character | characters], map_id, track_character) do
map_id
|> WandererApp.Map.Server.add_character(character, track_character)
:skip
end)
_add_characters(characters, map_id)
_add_characters(characters, map_id, track_character)
end
defp _remove_characters([], _map_id), do: :ok
@@ -2054,13 +2088,14 @@ defmodule WandererAppWeb.MapLive do
:ok = WandererApp.Character.TrackerManager.start_tracking(character_id)
end
defp _push_map_event(socket, type, event_body),
do:
defp _push_map_event(socket, type, event_body)
do
socket
|> push_event("map_event", %{
type: type,
body: event_body |> WandererApp.Utils.JSONUtil.compress()
})
end
defp map_id(%{assigns: %{map_id: map_id}} = _socket), do: map_id
end

View File

@@ -2,9 +2,10 @@
id="map-loader"
data-loading={show_loader("map-loader")}
data-loaded={hide_loader("map-loader")}
class="!z-100 absolute w-screen h-screen bg-transparent hidden"
class="!z-100 w-screen h-screen hidden relative"
>
<div class="flex w-full h-full items-center justify-center">
<div class="hs-overlay-backdrop transition duration absolute inset-0 blur" />
<div class="flex !z-[150] w-full h-full items-center justify-center">
<div class="Loader" data-text="Wanderer">
<span class="Loader__Circle"></span>
<span class="Loader__Circle"></span>
@@ -12,6 +13,8 @@
<span class="Loader__Circle"></span>
</div>
</div>
</div>
<div class="w-full h-full" id="mapper" phx-hook="Mapper" phx-update="ignore"></div>

View File

@@ -167,24 +167,6 @@ defmodule WandererAppWeb.MapsLive do
{:noreply, socket}
end
@impl true
def handle_event(
"live_select_change",
%{"id" => "form_owner_id_live_select_component" = id, "text" => text} = _change_event,
socket
) do
options =
if text == "" do
socket.assigns.characters
else
socket.assigns.characters
end
send_update(LiveSelect.Component, options: options, id: id)
{:noreply, socket}
end
@impl true
def handle_event(
"live_select_change",

View File

@@ -137,23 +137,17 @@
<.input type="text" field={f[:name]} placeholder="Name" />
<.input type="text" field={f[:slug]} prefix={@uri} placeholder="map-slug" />
<.input type="textarea" field={f[:description]} placeholder="Public description" />
<.live_select
<.input
type="select"
field={f[:owner_id]}
value_mapper={&map_character/1}
options={@characters}
class="p-dropdown p-component p-inputwrapper mt-8"
placeholder="Select a map owner"
>
<:option :let={option}>
<div class="flex items-center">
<.avatar url={member_icon_url(option.eve_id)} label={option.label} />
&nbsp;<%= option.label %>
</div>
</:option>
</.live_select>
options={Enum.map(@characters, fn character -> {character.label, character.id} end)}
/>
<.input
type="select"
field={f[:scope]}
class="p-dropdown p-component p-inputwrapper"
class="p-dropdown p-component p-inputwrapper mt-8"
placeholder="Select a map scope"
options={Enum.map(@scopes, fn scope -> {scope, scope} end)}
/>

View File

@@ -9,6 +9,10 @@ defmodule WandererAppWeb.Router do
warn: false,
only: [redirect_if_user_is_authenticated: 2]
import WandererAppWeb.BasicAuth,
warn: false,
only: [admin_basic_auth: 2]
@code_reloading Application.compile_env(
:wanderer_app,
[WandererAppWeb.Endpoint, :code_reloader],
@@ -20,6 +24,10 @@ defmodule WandererAppWeb.Router do
@font_src ~w('self' data: https://web.ccpgamescdn.com https://w.appzi.io)
@script_src ~w('self' )
pipeline :admin_bauth do
plug :admin_basic_auth
end
pipeline :browser do
plug(:accepts, ["html"])
plug(:fetch_session)
@@ -137,11 +145,9 @@ defmodule WandererAppWeb.Router do
get "/:provider/callback", AuthController, :callback
end
scope "/", WandererAppWeb do
scope "/admin", WandererAppWeb do
pipe_through(:browser)
get "/", RedirectController, :redirect_authenticated
get("/last", MapsController, :last)
pipe_through(:admin_bauth)
live_session :admin,
on_mount: [
@@ -149,9 +155,23 @@ defmodule WandererAppWeb.Router do
{WandererAppWeb.UserAuth, :ensure_admin},
WandererAppWeb.Nav
] do
live("/admin", AdminLive, :index)
live("/", AdminLive, :index)
end
error_tracker_dashboard("/errors",
on_mount: [
{WandererAppWeb.UserAuth, :ensure_authenticated},
{WandererAppWeb.UserAuth, :ensure_admin}
]
)
end
scope "/", WandererAppWeb do
pipe_through(:browser)
get "/", RedirectController, :redirect_authenticated
get("/last", MapsController, :last)
live_session :authenticated,
on_mount: [
{WandererAppWeb.UserAuth, :ensure_authenticated},
@@ -180,16 +200,7 @@ defmodule WandererAppWeb.Router do
end
end
scope "/admin" do
pipe_through(:browser)
error_tracker_dashboard("/errors",
on_mount: [
{WandererAppWeb.UserAuth, :ensure_authenticated},
{WandererAppWeb.UserAuth, :ensure_admin}
]
)
end
# Enable LiveDashboard and Swoosh mailbox preview in development
if Application.compile_env(:wanderer_app, :dev_routes) do
@@ -206,7 +217,6 @@ defmodule WandererAppWeb.Router do
error_tracker_dashboard("/errors", as: :error_tracker_dev_dashboard)
live_dashboard("/dashboard", metrics: WandererAppWeb.Telemetry)
# forward("/mailbox", Plug.Swoosh.MailboxPreview)
end
end
end

View File

@@ -2,7 +2,7 @@ defmodule WandererApp.MixProject do
use Mix.Project
@source_url "https://github.com/wanderer-industries/wanderer"
@version "1.0.20"
@version "1.2.5"
def project do
[