Files
pezkuwi-apps/packages/react-signer/src/Password.tsx
T

86 lines
2.1 KiB
TypeScript

// Copyright 2017-2026 @pezkuwi/react-signer authors & contributors
// SPDX-License-Identifier: Apache-2.0
import type { KeyringPair } from '@pezkuwi/keyring/types';
import React, { useEffect, useMemo, useState } from 'react';
import { Modal, Password, styled, Toggle } from '@pezkuwi/react-components';
import { keyring } from '@pezkuwi/ui-keyring';
import { useTranslation } from './translate.js';
import { UNLOCK_MINS } from './util.js';
interface Props {
address: string;
className?: string;
error?: string | null;
onChange: (password: string, isUnlockCached: boolean) => void;
onEnter?: () => void;
tabIndex?: number;
}
function getPair (address: string): KeyringPair | null {
try {
return keyring.getPair(address);
} catch {
return null;
}
}
function Unlock ({ address, className, error, onChange, onEnter, tabIndex }: Props): React.ReactElement<Props> | null {
const { t } = useTranslation();
const [password, setPassword] = useState('');
const [isUnlockCached, setIsUnlockCached] = useState(false);
const pair = useMemo(
() => getPair(address),
[address]
);
useEffect((): void => {
onChange(password, isUnlockCached);
}, [onChange, isUnlockCached, password]);
if (!pair || !pair.isLocked || pair.meta.isInjected) {
return null;
}
return (
<StyledModalColumns
className={className}
hint={t('Unlock the sending account to allow signing of this transaction.')}
>
<Password
autoFocus
isError={!!error}
label={t('unlock account with password')}
labelExtra={
<Toggle
label={t('unlock for {{expiry}} min', { replace: { expiry: UNLOCK_MINS } })}
onChange={setIsUnlockCached}
value={isUnlockCached}
/>
}
onChange={setPassword}
onEnter={onEnter}
tabIndex={tabIndex}
value={password}
/>
</StyledModalColumns>
);
}
const StyledModalColumns = styled(Modal.Columns)`
.errorLabel {
margin-right: 1rem;
color: #9f3a38 !important;
}
.ui--Toggle {
bottom: 1.1rem;
}
`;
export default React.memo(Unlock);