|
|
@ -8,12 +8,7 @@ import Electron from 'electron';
|
|
|
|
const { shell } = Electron;
|
|
|
|
const { shell } = Electron;
|
|
|
|
|
|
|
|
|
|
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
|
|
import {
|
|
|
|
import { SessionIcon, SessionIconButton, SessionIconSize, SessionIconType } from './session/icon';
|
|
|
|
SessionIcon,
|
|
|
|
|
|
|
|
SessionIconButton,
|
|
|
|
|
|
|
|
SessionIconSize,
|
|
|
|
|
|
|
|
SessionIconType,
|
|
|
|
|
|
|
|
} from './session/icon';
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
import { SessionWrapperModal } from './session/SessionWrapperModal';
|
|
|
|
import { SessionWrapperModal } from './session/SessionWrapperModal';
|
|
|
|
|
|
|
|
|
|
|
@ -31,6 +26,7 @@ import {
|
|
|
|
// tslint:disable-next-line: no-submodule-imports
|
|
|
|
// tslint:disable-next-line: no-submodule-imports
|
|
|
|
import useNetworkState from 'react-use/lib/useNetworkState';
|
|
|
|
import useNetworkState from 'react-use/lib/useNetworkState';
|
|
|
|
import { SessionSpinner } from './session/SessionSpinner';
|
|
|
|
import { SessionSpinner } from './session/SessionSpinner';
|
|
|
|
|
|
|
|
import { Flex } from './basic/Flex';
|
|
|
|
|
|
|
|
|
|
|
|
export type StatusLightType = {
|
|
|
|
export type StatusLightType = {
|
|
|
|
glowStartDelay: number;
|
|
|
|
glowStartDelay: number;
|
|
|
@ -60,56 +56,57 @@ const OnionPathModalInner = () => {
|
|
|
|
<>
|
|
|
|
<>
|
|
|
|
<p className="onion__description">{window.i18n('onionPathIndicatorDescription')}</p>
|
|
|
|
<p className="onion__description">{window.i18n('onionPathIndicatorDescription')}</p>
|
|
|
|
<div className="onion__node-list">
|
|
|
|
<div className="onion__node-list">
|
|
|
|
<div className="onion__vertical-line" />
|
|
|
|
<Flex container={true}>
|
|
|
|
{nodes.map((snode: Snode | any, index: number) => {
|
|
|
|
<div className="onion__node-list-lights">
|
|
|
|
return (
|
|
|
|
<div className="onion__vertical-line" />
|
|
|
|
<>
|
|
|
|
|
|
|
|
<OnionNodeStatusLight
|
|
|
|
<Flex container={true} flexDirection="column" alignItems="center" height="100%">
|
|
|
|
glowDuration={glowDuration}
|
|
|
|
{nodes.map((_snode: Snode | any, index: number) => {
|
|
|
|
glowStartDelay={index}
|
|
|
|
return (
|
|
|
|
label={snode.label}
|
|
|
|
<OnionNodeStatusLight
|
|
|
|
snode={snode}
|
|
|
|
glowDuration={glowDuration}
|
|
|
|
key={index}
|
|
|
|
glowStartDelay={index}
|
|
|
|
/>
|
|
|
|
key={index}
|
|
|
|
</>
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
})}
|
|
|
|
|
|
|
|
</Flex>
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<Flex container={true} flexDirection="column" alignItems="flex-start">
|
|
|
|
|
|
|
|
{nodes.map((snode: Snode | any, index: number) => {
|
|
|
|
|
|
|
|
let labelText = snode.label
|
|
|
|
|
|
|
|
? snode.label
|
|
|
|
|
|
|
|
: countryLookup.byIso(ip2country(snode.ip))?.country;
|
|
|
|
|
|
|
|
if (!labelText) {
|
|
|
|
|
|
|
|
labelText = window.i18n('unknownCountry');
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return labelText ? <div className="onion__node__country">{labelText}</div> : null;
|
|
|
|
|
|
|
|
})}
|
|
|
|
|
|
|
|
</Flex>
|
|
|
|
|
|
|
|
</Flex>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
export type OnionNodeStatusLightType = {
|
|
|
|
export type OnionNodeStatusLightType = {
|
|
|
|
snode: Snode;
|
|
|
|
|
|
|
|
label?: string;
|
|
|
|
|
|
|
|
glowStartDelay: number;
|
|
|
|
glowStartDelay: number;
|
|
|
|
glowDuration: number;
|
|
|
|
glowDuration: number;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
/**
|
|
|
|
* Component containing a coloured status light and an adjacent country label.
|
|
|
|
* Component containing a coloured status light.
|
|
|
|
*/
|
|
|
|
*/
|
|
|
|
export const OnionNodeStatusLight = (props: OnionNodeStatusLightType): JSX.Element => {
|
|
|
|
export const OnionNodeStatusLight = (props: OnionNodeStatusLightType): JSX.Element => {
|
|
|
|
const { snode, label, glowStartDelay, glowDuration } = props;
|
|
|
|
const { glowStartDelay, glowDuration } = props;
|
|
|
|
const theme = useTheme();
|
|
|
|
const theme = useTheme();
|
|
|
|
|
|
|
|
|
|
|
|
let labelText = label ? label : countryLookup.byIso(ip2country(snode.ip))?.country;
|
|
|
|
|
|
|
|
if (!labelText) {
|
|
|
|
|
|
|
|
labelText = window.i18n('unknownCountry');
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
return (
|
|
|
|
<div className="onion__node">
|
|
|
|
<ModalStatusLight
|
|
|
|
<ModalStatusLight
|
|
|
|
glowDuration={glowDuration}
|
|
|
|
glowDuration={glowDuration}
|
|
|
|
glowStartDelay={glowStartDelay}
|
|
|
|
glowStartDelay={glowStartDelay}
|
|
|
|
color={theme.colors.accent}
|
|
|
|
color={theme.colors.accent}
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
|
|
|
|
{labelText ? (
|
|
|
|
|
|
|
|
<>
|
|
|
|
|
|
|
|
<div className="onion__node__country">{labelText}</div>
|
|
|
|
|
|
|
|
</>
|
|
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
);
|
|
|
|
);
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
@ -128,7 +125,7 @@ export const ModalStatusLight = (props: StatusLightType) => {
|
|
|
|
glowDuration={glowDuration}
|
|
|
|
glowDuration={glowDuration}
|
|
|
|
glowStartDelay={glowStartDelay}
|
|
|
|
glowStartDelay={glowStartDelay}
|
|
|
|
iconType={SessionIconType.Circle}
|
|
|
|
iconType={SessionIconType.Circle}
|
|
|
|
iconSize={SessionIconSize.Small}
|
|
|
|
iconSize={SessionIconSize.Tiny}
|
|
|
|
theme={theme}
|
|
|
|
theme={theme}
|
|
|
|
/>
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
@ -163,7 +160,7 @@ export const ActionPanelOnionStatusLight = (props: {
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
return (
|
|
|
|
<SessionIconButton
|
|
|
|
<SessionIconButton
|
|
|
|
iconSize={SessionIconSize.Medium}
|
|
|
|
iconSize={SessionIconSize.Small}
|
|
|
|
iconType={SessionIconType.Circle}
|
|
|
|
iconType={SessionIconType.Circle}
|
|
|
|
iconColor={iconColor}
|
|
|
|
iconColor={iconColor}
|
|
|
|
onClick={handleClick}
|
|
|
|
onClick={handleClick}
|
|
|
|