import * as React from 'react'; import { sprintf } from 'sprintf-js'; import { TunnelProtocol } from '../../shared/daemon-rpc-types'; import { messages } from '../../shared/gettext'; import { WgKeyState } from '../redux/settings/reducers'; import { StyledNavigationScrollbars, StyledNoWireguardKeyError, StyledNoWireguardKeyErrorContainer, StyledSelectorForFooter, StyledTunnelProtocolContainer, } from './AdvancedSettingsStyles'; import * as AppButton from './AppButton'; import { AriaDescription, AriaInput, AriaInputGroup, AriaLabel } from './AriaGroup'; import * as Cell from './cell'; import CustomDnsSettings from './CustomDnsSettings'; import { Layout, SettingsContainer } from './Layout'; import { ModalAlert, ModalAlertType, ModalContainer, ModalMessage } from './Modal'; import { BackBarItem, NavigationBar, NavigationContainer, NavigationItems, TitleBarItem, } from './NavigationBar'; import { ISelectorItem } from './cell/Selector'; import SettingsHeader, { HeaderTitle } from './SettingsHeader'; import Switch from './Switch'; type OptionalTunnelProtocol = TunnelProtocol | undefined; interface IProps { enableIpv6: boolean; blockWhenDisconnected: boolean; tunnelProtocol?: TunnelProtocol; wireguardKeyState: WgKeyState; setEnableIpv6: (value: boolean) => void; setBlockWhenDisconnected: (value: boolean) => void; setTunnelProtocol: (value: OptionalTunnelProtocol) => void; onViewWireguardSettings: () => void; onViewOpenVpnSettings: () => void; onViewSplitTunneling: () => void; onClose: () => void; } interface IState { showConfirmBlockWhenDisconnectedAlert: boolean; } export default class AdvancedSettings extends React.Component { public state = { showConfirmBlockWhenDisconnectedAlert: false, }; private blockWhenDisconnectedRef = React.createRef(); public render() { const hasWireguardKey = this.props.wireguardKeyState.type === 'key-set'; return ( { // TRANSLATORS: Back button in navigation bar messages.pgettext('navigation-bar', 'Settings') } { // TRANSLATORS: Title label in navigation bar messages.pgettext('advanced-settings-nav', 'Advanced') } {messages.pgettext('advanced-settings-view', 'Advanced')} {messages.pgettext('advanced-settings-view', 'Enable IPv6')} {messages.pgettext( 'advanced-settings-view', 'Enable IPv6 communication through the tunnel.', )} {messages.pgettext('advanced-settings-view', 'Always require VPN')} {messages.pgettext( 'advanced-settings-view', 'If you disconnect or quit the app, this setting will block your internet.', )} {(window.env.platform === 'linux' || window.env.platform === 'win32') && ( {messages.pgettext('advanced-settings-view', 'Split tunneling')} )} {!hasWireguardKey && ( {messages.pgettext( 'advanced-settings-view', 'To enable WireGuard, generate a key under the "WireGuard key" setting below.', )} )} {messages.pgettext('advanced-settings-view', 'WireGuard settings')} {messages.pgettext('advanced-settings-view', 'OpenVPN settings')} {this.state.showConfirmBlockWhenDisconnectedAlert && this.renderConfirmBlockWhenDisconnectedAlert()} ); } private tunnelProtocolItems = ( hasWireguardKey: boolean, ): Array> => { return [ { label: messages.gettext('Automatic'), value: undefined, }, { label: hasWireguardKey ? messages.pgettext('advanced-settings-view', 'WireGuard') : sprintf('%(label)s (%(error)s)', { label: messages.pgettext('advanced-settings-view', 'WireGuard'), error: messages.pgettext('advanced-settings-view', 'missing key'), }), value: 'wireguard', disabled: !hasWireguardKey, }, { label: messages.pgettext('advanced-settings-view', 'OpenVPN'), value: 'openvpn', }, ]; }; private renderConfirmBlockWhenDisconnectedAlert = () => { return ( {messages.pgettext('advanced-settings-view', 'Enable anyway')} , {messages.gettext('Back')} , ]} close={this.hideConfirmBlockWhenDisconnectedAlert}> {messages.pgettext( 'advanced-settings-view', 'Attention: enabling this will always require a Mullvad VPN connection in order to reach the internet.', )} {messages.pgettext( 'advanced-settings-view', 'The app’s built-in kill switch is always on. This setting will additionally block the internet if clicking Disconnect or Quit.', )} ); }; private setBlockWhenDisconnected = (newValue: boolean) => { if (newValue) { this.setState({ showConfirmBlockWhenDisconnectedAlert: true }); } else { this.props.setBlockWhenDisconnected(false); } }; private hideConfirmBlockWhenDisconnectedAlert = () => { this.setState({ showConfirmBlockWhenDisconnectedAlert: false }); this.blockWhenDisconnectedRef.current?.setOn(this.props.blockWhenDisconnected); }; private confirmEnableBlockWhenDisconnected = () => { this.setState({ showConfirmBlockWhenDisconnectedAlert: false }); this.props.setBlockWhenDisconnected(true); }; private onSelectTunnelProtocol = (protocol?: TunnelProtocol) => { this.props.setTunnelProtocol(protocol); }; }