You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
205 lines
5.6 KiB
TypeScript
205 lines
5.6 KiB
TypeScript
import React from 'react';
|
|
|
|
import { shell } from 'electron';
|
|
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
|
|
import ip2country from 'ip2country';
|
|
import countryLookup from 'country-code-lookup';
|
|
import { Snode } from '../../data/data';
|
|
import { onionPathModal } from '../../state/ducks/modalDialog';
|
|
import {
|
|
getFirstOnionPath,
|
|
getFirstOnionPathLength,
|
|
getIsOnline,
|
|
getOnionPathsCount,
|
|
} from '../../state/selectors/onions';
|
|
import { Flex } from '../basic/Flex';
|
|
import { SessionIcon, SessionIconButton } from '../session/icon';
|
|
import { SessionSpinner } from '../session/SessionSpinner';
|
|
import { SessionWrapperModal } from '../session/SessionWrapperModal';
|
|
// tslint:disable-next-line: no-submodule-imports
|
|
import useHover from 'react-use/lib/useHover';
|
|
|
|
export type StatusLightType = {
|
|
glowStartDelay: number;
|
|
glowDuration: number;
|
|
color?: string;
|
|
};
|
|
|
|
const OnionCountryDisplay = ({
|
|
index,
|
|
labelText,
|
|
snodeIp,
|
|
}: {
|
|
snodeIp?: string;
|
|
labelText: string;
|
|
index: number;
|
|
}) => {
|
|
const element = (hovered: boolean) => (
|
|
<div className="onion__node__country" key={`country-${index}`}>
|
|
{hovered && snodeIp ? snodeIp : labelText}
|
|
</div>
|
|
);
|
|
const [hoverable] = useHover(element);
|
|
|
|
return hoverable;
|
|
};
|
|
|
|
const OnionPathModalInner = () => {
|
|
const onionPath = useSelector(getFirstOnionPath);
|
|
const isOnline = useSelector(getIsOnline);
|
|
// including the device and destination in calculation
|
|
const glowDuration = onionPath.length + 2;
|
|
if (!isOnline || !onionPath || onionPath.length === 0) {
|
|
return <SessionSpinner loading={true} />;
|
|
}
|
|
|
|
const nodes = [
|
|
{
|
|
label: window.i18n('device'),
|
|
},
|
|
...onionPath,
|
|
{
|
|
label: window.i18n('destination'),
|
|
},
|
|
];
|
|
|
|
return (
|
|
<>
|
|
<p className="onion__description">{window.i18n('onionPathIndicatorDescription')}</p>
|
|
<div className="onion__node-list">
|
|
<Flex container={true}>
|
|
<div className="onion__node-list-lights">
|
|
<div className="onion__vertical-line" />
|
|
|
|
<Flex container={true} flexDirection="column" alignItems="center" height="100%">
|
|
{nodes.map((_snode: Snode | any, index: number) => {
|
|
return (
|
|
<OnionNodeStatusLight
|
|
glowDuration={glowDuration}
|
|
glowStartDelay={index}
|
|
key={`light-${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 ? (
|
|
<OnionCountryDisplay index={index} labelText={labelText} snodeIp={snode.ip} />
|
|
) : null;
|
|
})}
|
|
</Flex>
|
|
</Flex>
|
|
</div>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export type OnionNodeStatusLightType = {
|
|
glowStartDelay: number;
|
|
glowDuration: number;
|
|
};
|
|
|
|
/**
|
|
* Component containing a coloured status light.
|
|
*/
|
|
export const OnionNodeStatusLight = (props: OnionNodeStatusLightType): JSX.Element => {
|
|
const { glowStartDelay, glowDuration } = props;
|
|
|
|
return (
|
|
<ModalStatusLight
|
|
glowDuration={glowDuration}
|
|
glowStartDelay={glowStartDelay}
|
|
color={'var(--color-accent)'}
|
|
/>
|
|
);
|
|
};
|
|
|
|
/**
|
|
* An icon with a pulsating glow emission.
|
|
*/
|
|
export const ModalStatusLight = (props: StatusLightType) => {
|
|
const { glowStartDelay, glowDuration, color } = props;
|
|
|
|
return (
|
|
<div className="onion__growing-icon">
|
|
<SessionIcon
|
|
borderRadius={'50px'}
|
|
iconColor={color}
|
|
glowDuration={glowDuration}
|
|
glowStartDelay={glowStartDelay}
|
|
iconType="circle"
|
|
iconSize={'tiny'}
|
|
/>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
/**
|
|
* A status light specifically for the action panel. Color is based on aggregate node states instead of individual onion node state
|
|
*/
|
|
export const ActionPanelOnionStatusLight = (props: {
|
|
isSelected: boolean;
|
|
handleClick: () => void;
|
|
}) => {
|
|
const { isSelected, handleClick } = props;
|
|
|
|
const onionPathsCount = useSelector(getOnionPathsCount);
|
|
const firstPathLength = useSelector(getFirstOnionPathLength);
|
|
const isOnline = useSelector(getIsOnline);
|
|
|
|
// Set icon color based on result
|
|
const red = 'var(--color-destructive)';
|
|
const green = 'var(--color-accent)';
|
|
const orange = 'var(--color-warning)';
|
|
|
|
// start with red
|
|
let iconColor = red;
|
|
//if we are not online or the first path is not valid, we keep red as color
|
|
if (isOnline && firstPathLength > 1) {
|
|
iconColor = onionPathsCount >= 2 ? green : onionPathsCount >= 1 ? orange : red;
|
|
}
|
|
|
|
return (
|
|
<SessionIconButton
|
|
iconSize={'small'}
|
|
iconType="circle"
|
|
iconColor={iconColor}
|
|
onClick={handleClick}
|
|
glowDuration={10}
|
|
glowStartDelay={0}
|
|
noScale={true}
|
|
isSelected={isSelected}
|
|
/>
|
|
);
|
|
};
|
|
|
|
export const OnionPathModal = () => {
|
|
const onConfirm = () => {
|
|
void shell.openExternal('https://getsession.org/faq/#onion-routing');
|
|
};
|
|
const dispatch = useDispatch();
|
|
return (
|
|
// tslint:disable-next-line: use-simple-attributes
|
|
<SessionWrapperModal
|
|
title={window.i18n('onionPathIndicatorTitle')}
|
|
confirmText={window.i18n('learnMore')}
|
|
cancelText={window.i18n('cancel')}
|
|
onConfirm={onConfirm}
|
|
onClose={() => dispatch(onionPathModal(null))}
|
|
showExitIcon={true}
|
|
>
|
|
<OnionPathModalInner />
|
|
</SessionWrapperModal>
|
|
);
|
|
};
|