import { setWallet, TSetWallet, unlockKeystore, TUnlockKeystore, unlockMnemonic, TUnlockMnemonic, unlockPrivateKey, TUnlockPrivateKey, unlockWeb3, TUnlockWeb3, resetWallet, TResetWallet } from 'actions/wallet'; import { reset, TReset } from 'actions/transaction'; import isEmpty from 'lodash/isEmpty'; import map from 'lodash/map'; import React, { Component } from 'react'; import { connect } from 'react-redux'; import translate from 'translations'; import KeystoreDecrypt from './Keystore'; import LedgerNanoSDecrypt from './LedgerNano'; import MnemonicDecrypt from './Mnemonic'; import PrivateKeyDecrypt, { PrivateKeyValue } from './PrivateKey'; import TrezorDecrypt from './Trezor'; import ViewOnlyDecrypt from './ViewOnly'; import { AppState } from 'reducers'; import Web3Decrypt from './Web3'; import Help from 'components/ui/Help'; import { knowledgeBaseURL } from 'config/data'; import NavigationPrompt from './NavigationPrompt'; import { IWallet } from 'libs/wallet'; import { showNotification, TShowNotification } from 'actions/notifications'; type UnlockParams = {} | PrivateKeyValue; interface Props { resetTransactionState: TReset; unlockKeystore: TUnlockKeystore; unlockMnemonic: TUnlockMnemonic; unlockPrivateKey: TUnlockPrivateKey; setWallet: TSetWallet; unlockWeb3: TUnlockWeb3; resetWallet: TResetWallet; showNotification: TShowNotification; wallet: IWallet; hidden?: boolean; offline: boolean; allowReadOnly?: boolean; disabledWallets?: string[]; } interface State { selectedWalletKey: string; value: UnlockParams; } export class WalletDecrypt extends Component { public WALLETS = { web3: { lid: 'x_MetaMask', component: Web3Decrypt, initialParams: {}, unlock: this.props.unlockWeb3, helpLink: `${knowledgeBaseURL}/migration/moving-from-private-key-to-metamask` }, 'ledger-nano-s': { lid: 'x_Ledger', component: LedgerNanoSDecrypt, initialParams: {}, unlock: this.props.setWallet, helpLink: 'https://ledger.zendesk.com/hc/en-us/articles/115005200009-How-to-use-MyEtherWallet-with-Ledger' }, trezor: { lid: 'x_Trezor', component: TrezorDecrypt, initialParams: {}, unlock: this.props.setWallet, helpLink: 'https://doc.satoshilabs.com/trezor-apps/mew.html' }, 'keystore-file': { lid: 'x_Keystore2', component: KeystoreDecrypt, initialParams: { file: '', password: '' }, unlock: this.props.unlockKeystore, helpLink: `${knowledgeBaseURL}/private-keys-passwords/difference-beween-private-key-and-keystore-file.html` }, 'mnemonic-phrase': { lid: 'x_Mnemonic', component: MnemonicDecrypt, initialParams: {}, unlock: this.props.unlockMnemonic, helpLink: `${knowledgeBaseURL}/private-keys-passwords/difference-beween-private-key-and-keystore-file.html` }, 'private-key': { lid: 'x_PrivKey2', component: PrivateKeyDecrypt, initialParams: { key: '', password: '' }, unlock: this.props.unlockPrivateKey, helpLink: `${knowledgeBaseURL}/private-keys-passwords/difference-beween-private-key-and-keystore-file.html` }, 'view-only': { lid: 'View with Address Only', component: ViewOnlyDecrypt, initialParams: {}, unlock: this.props.setWallet, helpLink: '' } }; public state: State = { selectedWalletKey: 'keystore-file', value: this.WALLETS['keystore-file'].initialParams }; public getDecryptionComponent() { const { selectedWalletKey, value } = this.state; const selectedWallet = this.WALLETS[selectedWalletKey]; if (!selectedWallet) { return null; } return ( ); } public isOnlineRequiredWalletAndOffline(selectedWalletKey) { const onlineRequiredWallets = ['trezor', 'ledger-nano-s']; return this.props.offline && onlineRequiredWallets.includes(selectedWalletKey); } public buildWalletOptions() { return map(this.WALLETS, (wallet, key) => { const { helpLink } = wallet; const isSelected = this.state.selectedWalletKey === key; const isDisabled = this.isOnlineRequiredWalletAndOffline(key) || (!this.props.allowReadOnly && wallet.component === ViewOnlyDecrypt) || this.isWalletDisabled(key); return ( ); }); } public handleDecryptionChoiceChange = (event: React.FormEvent) => { const wallet = this.WALLETS[event.currentTarget.value]; if (!wallet) { return; } this.setState({ selectedWalletKey: event.currentTarget.value, value: wallet.initialParams }); }; public render() { const { wallet, hidden } = this.props; const decryptionComponent = this.getDecryptionComponent(); const unlocked = !!wallet; return (
{!hidden && (

{translate('decrypt_Access')}

{this.buildWalletOptions()}
{decryptionComponent} {!!(this.state.value as PrivateKeyValue).valid && (

{translate('ADD_Label_6')}

)}
)}
); } public onChange = (value: UnlockParams) => { this.setState({ value }); }; public onUnlock = (payload: any) => { // some components (TrezorDecrypt) don't take an onChange prop, and thus this.state.value will remain unpopulated. // in this case, we can expect the payload to contain the unlocked wallet info. const unlockValue = this.state.value && !isEmpty(this.state.value) ? this.state.value : payload; this.WALLETS[this.state.selectedWalletKey].unlock(unlockValue); this.props.resetTransactionState(); }; private isWalletDisabled = (walletKey: string) => { if (!this.props.disabledWallets) { return false; } return this.props.disabledWallets.indexOf(walletKey) !== -1; }; } function mapStateToProps(state: AppState) { return { offline: state.config.offline, wallet: state.wallet.inst }; } export default connect(mapStateToProps, { unlockKeystore, unlockMnemonic, unlockPrivateKey, unlockWeb3, setWallet, resetWallet, resetTransactionState: reset, showNotification })(WalletDecrypt);