123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181 |
- import {Fragment, useState} from 'react';
- import styled from '@emotion/styled';
- import {Button} from 'sentry/components/button';
- import Confirm from 'sentry/components/confirm';
- import DateTime from 'sentry/components/dateTime';
- import EmptyMessage from 'sentry/components/emptyMessage';
- import Input from 'sentry/components/input';
- import {Panel, PanelBody, PanelHeader, PanelItem} from 'sentry/components/panels';
- import {Tooltip} from 'sentry/components/tooltip';
- import {IconClose, IconDelete} from 'sentry/icons';
- import {t} from 'sentry/locale';
- import {space} from 'sentry/styles/space';
- import ConfirmHeader from 'sentry/views/settings/account/accountSecurity/components/confirmHeader';
- import TextBlock from 'sentry/views/settings/components/text/textBlock';
- function U2fEnrolledDetails(props) {
- const {className, isEnrolled, devices, id, onRemoveU2fDevice, onRenameU2fDevice} =
- props;
- if (id !== 'u2f' || !isEnrolled) {
- return null;
- }
- const hasDevices = devices?.length;
- // Note Tooltip doesn't work because of bootstrap(?) pointer events for disabled buttons
- const isLastDevice = hasDevices === 1;
- return (
- <Panel className={className}>
- <PanelHeader>{t('Device name')}</PanelHeader>
- <PanelBody>
- {!hasDevices && (
- <EmptyMessage>{t('You have not added any U2F devices')}</EmptyMessage>
- )}
- {hasDevices &&
- devices?.map((device, i) => (
- <Device
- key={i}
- device={device}
- isLastDevice={isLastDevice}
- onRenameU2fDevice={onRenameU2fDevice}
- onRemoveU2fDevice={onRemoveU2fDevice}
- />
- ))}
- <AddAnotherPanelItem>
- <Button to="/settings/account/security/mfa/u2f/enroll/" size="sm">
- {t('Add Another Device')}
- </Button>
- </AddAnotherPanelItem>
- </PanelBody>
- </Panel>
- );
- }
- function Device(props) {
- const {device, isLastDevice, onRenameU2fDevice, onRemoveU2fDevice} = props;
- const [deviceName, setDeviceName] = useState(device.name);
- const [isEditing, setEditting] = useState(false);
- if (!isEditing) {
- return (
- <DevicePanelItem key={device.name}>
- <DeviceInformation>
- <Name>{device.name}</Name>
- <FadedDateTime date={device.timestamp} />
- </DeviceInformation>
- <Actions>
- <Button size="sm" onClick={() => setEditting(true)}>
- {t('Rename Device')}
- </Button>
- </Actions>
- <Actions>
- <Confirm
- onConfirm={() => onRemoveU2fDevice(device)}
- disabled={isLastDevice}
- message={
- <Fragment>
- <ConfirmHeader>{t('Do you want to remove U2F device?')}</ConfirmHeader>
- <TextBlock>
- {t('Are you sure you want to remove the U2F device "%s"?', device.name)}
- </TextBlock>
- </Fragment>
- }
- >
- <Button size="sm" priority="danger">
- <Tooltip
- disabled={!isLastDevice}
- title={t('Can not remove last U2F device')}
- >
- <IconDelete size="xs" />
- </Tooltip>
- </Button>
- </Confirm>
- </Actions>
- </DevicePanelItem>
- );
- }
- return (
- <DevicePanelItem key={device.name}>
- <DeviceInformation>
- <DeviceNameInput
- type="text"
- value={deviceName}
- onChange={e => {
- setDeviceName(e.target.value);
- }}
- />
- <FadedDateTime date={device.timestamp} />
- </DeviceInformation>
- <Actions>
- <Button
- priority="primary"
- size="sm"
- onClick={() => {
- onRenameU2fDevice(device, deviceName);
- setEditting(false);
- }}
- >
- Rename Device
- </Button>
- </Actions>
- <Actions>
- <Button
- size="sm"
- title="Cancel rename"
- onClick={() => {
- setDeviceName(device.name);
- setEditting(false);
- }}
- >
- <IconClose size="xs" />
- </Button>
- </Actions>
- </DevicePanelItem>
- );
- }
- const DeviceNameInput = styled(Input)`
- width: 50%;
- margin-right: ${space(2)};
- `;
- const DevicePanelItem = styled(PanelItem)`
- padding: 0;
- `;
- const DeviceInformation = styled('div')`
- display: flex;
- align-items: center;
- justify-content: space-between;
- flex: 1 1;
- height: 72px;
- padding: ${space(2)};
- padding-right: 0;
- `;
- const FadedDateTime = styled(DateTime)`
- font-size: ${p => p.theme.fontSizeRelativeSmall};
- opacity: 0.6;
- `;
- const Name = styled('div')`
- flex: 1;
- `;
- const Actions = styled('div')`
- margin: ${space(2)};
- `;
- const AddAnotherPanelItem = styled(PanelItem)`
- justify-content: flex-end;
- padding: ${space(2)};
- `;
- export default styled(U2fEnrolledDetails)`
- margin-top: ${space(4)};
- `;
|