From a57d17a3e0f91ce44efe10bbc375d772d2d391b3 Mon Sep 17 00:00:00 2001 From: William O'Beirne Date: Mon, 30 Apr 2018 14:39:25 -0400 Subject: [PATCH 01/39] Fix Duplicate DPaths & Deterministic Modal Improvements (#1682) * Fix dPath uniqueness * Pass around dPath object, not just path, to maintain uniqueness * Improve deterministic modal UI * Fix custom dpaths --- .../components/DeterministicWalletsModal.scss | 14 ++++-- .../components/DeterministicWalletsModal.tsx | 50 +++++++++---------- .../WalletDecrypt/components/LedgerNano.tsx | 20 ++++---- .../WalletDecrypt/components/Mnemonic.tsx | 10 ++-- .../WalletDecrypt/components/Trezor.tsx | 22 ++++---- common/selectors/config/wallet.ts | 4 +- 6 files changed, 62 insertions(+), 58 deletions(-) diff --git a/common/components/WalletDecrypt/components/DeterministicWalletsModal.scss b/common/components/WalletDecrypt/components/DeterministicWalletsModal.scss index cbc474ee..88816a5f 100644 --- a/common/components/WalletDecrypt/components/DeterministicWalletsModal.scss +++ b/common/components/WalletDecrypt/components/DeterministicWalletsModal.scss @@ -46,7 +46,7 @@ } &-addresses { - overflow-y: scroll; + overflow-y: auto; &-table { width: 732px; text-align: center; @@ -74,13 +74,19 @@ background-image: url('~assets/images/icon-external-link.svg'); } + &-na { + font-size: $font-size-xs; + opacity: 0.3; + } + + // Specific selectors to override bootstrap tbody { tr { cursor: pointer; - } - td { - vertical-align: middle; + td { + vertical-align: middle; + } } } } diff --git a/common/components/WalletDecrypt/components/DeterministicWalletsModal.tsx b/common/components/WalletDecrypt/components/DeterministicWalletsModal.tsx index ee575938..80ba7135 100644 --- a/common/components/WalletDecrypt/components/DeterministicWalletsModal.tsx +++ b/common/components/WalletDecrypt/components/DeterministicWalletsModal.tsx @@ -24,7 +24,7 @@ const WALLETS_PER_PAGE = 5; interface Props { // Passed props isOpen?: boolean; - dPath: string; + dPath: DPath; dPaths: DPath[]; publicKey?: string; chainCode?: string; @@ -42,11 +42,11 @@ interface Props { onCancel(): void; onConfirmAddress(address: string, addressIndex: number): void; - onPathChange(path: string): void; + onPathChange(dPath: DPath): void; } interface State { - currentLabel: string; + currentDPath: DPath; selectedAddress: string; selectedAddrIndex: number; isCustomPath: boolean; @@ -65,7 +65,7 @@ class DeterministicWalletsModalClass extends React.PureComponent { selectedAddrIndex: 0, isCustomPath: false, customPath: '', - currentLabel: '', + currentDPath: this.props.dPath, page: 0 }; @@ -86,7 +86,7 @@ class DeterministicWalletsModalClass extends React.PureComponent { } public render() { - const { wallets, desiredToken, network, tokens, dPath, dPaths, onCancel } = this.props; + const { wallets, desiredToken, network, tokens, dPaths, onCancel } = this.props; const { selectedAddress, customPath, page } = this.state; const buttons: IButton[] = [ @@ -119,7 +119,7 @@ class DeterministicWalletsModalClass extends React.PureComponent {
{ const { dPath, publicKey, chainCode, seed } = props; if (dPath && ((publicKey && chainCode) || seed)) { - if (isValidPath(dPath)) { + if (isValidPath(dPath.value)) { this.props.getDeterministicWallets({ seed, - dPath, publicKey, chainCode, + dPath: dPath.value, limit: WALLETS_PER_PAGE, offset: WALLETS_PER_PAGE * this.state.page }); @@ -214,19 +214,12 @@ class DeterministicWalletsModalClass extends React.PureComponent { } } - private findDPath = (prop: keyof DPath, cmp: string) => { - return this.props.dPaths.find(d => d[prop] === cmp) || customDPath; - }; - private handleChangePath = (newPath: DPath) => { - const { value: dPathLabel } = newPath; - const { value } = this.findDPath('value', dPathLabel); - - if (value === customDPath.value) { - this.setState({ isCustomPath: true, currentLabel: dPathLabel }); + if (newPath.value === customDPath.value) { + this.setState({ isCustomPath: true, currentDPath: newPath }); } else { - this.setState({ isCustomPath: false, currentLabel: dPathLabel }); - this.props.onPathChange(value); + this.setState({ isCustomPath: false, currentDPath: newPath }); + this.props.onPathChange(newPath); } }; @@ -235,11 +228,14 @@ class DeterministicWalletsModalClass extends React.PureComponent { }; private handleSubmitCustomPath = (ev: React.FormEvent) => { - const { customPath, currentLabel } = this.state; + const { customPath, currentDPath } = this.state; ev.preventDefault(); - if (currentLabel === customDPath.label && isValidPath(customPath)) { - this.props.onPathChange(customPath); + if (currentDPath.value === customDPath.value && isValidPath(customPath)) { + this.props.onPathChange({ + label: customDPath.label, + value: customPath + }); } }; @@ -309,16 +305,16 @@ class DeterministicWalletsModalClass extends React.PureComponent { /> - {token ? ( + {desiredToken ? ( ) : ( - '???' + N/A )} diff --git a/common/components/WalletDecrypt/components/LedgerNano.tsx b/common/components/WalletDecrypt/components/LedgerNano.tsx index 06c36e0c..e4b3e222 100644 --- a/common/components/WalletDecrypt/components/LedgerNano.tsx +++ b/common/components/WalletDecrypt/components/LedgerNano.tsx @@ -26,7 +26,7 @@ interface StateProps { interface State { publicKey: string; chainCode: string; - dPath: string; + dPath: DPath; error: string | null; isLoading: boolean; showTip: boolean; @@ -38,7 +38,7 @@ class LedgerNanoSDecryptClass extends PureComponent { public state: State = { publicKey: '', chainCode: '', - dPath: this.props.dPath ? this.props.dPath.value : '', + dPath: this.props.dPath || this.props.dPaths[0], error: null, isLoading: false, showTip: false @@ -51,8 +51,8 @@ class LedgerNanoSDecryptClass extends PureComponent { }; public componentWillReceiveProps(nextProps: Props) { - if (this.props.dPath !== nextProps.dPath) { - this.setState({ dPath: nextProps.dPath ? nextProps.dPath.value : '' }); + if (this.props.dPath !== nextProps.dPath && nextProps.dPath) { + this.setState({ dPath: nextProps.dPath }); } } @@ -123,11 +123,11 @@ class LedgerNanoSDecryptClass extends PureComponent { ); } - private handlePathChange = (dPath: string) => { + private handlePathChange = (dPath: DPath) => { this.handleConnect(dPath); }; - private handleConnect = (dPath: string = this.state.dPath) => { + private handleConnect = (dPath: DPath) => { this.setState({ isLoading: true, error: null, @@ -136,7 +136,7 @@ class LedgerNanoSDecryptClass extends PureComponent { ledger.comm_u2f.create_async().then((comm: any) => { new ledger.eth(comm) - .getAddress_async(dPath, false, true) + .getAddress_async(dPath.value, false, true) .then(res => { this.setState({ publicKey: res.publicKey, @@ -182,19 +182,19 @@ class LedgerNanoSDecryptClass extends PureComponent { }; private handleUnlock = (address: string, index: number) => { - this.props.onUnlock(new LedgerWallet(address, this.state.dPath, index)); + this.props.onUnlock(new LedgerWallet(address, this.state.dPath.value, index)); this.reset(); }; private handleNullConnect = (): void => { - return this.handleConnect(); + return this.handleConnect(this.state.dPath); }; private reset() { this.setState({ publicKey: '', chainCode: '', - dPath: this.props.dPath ? this.props.dPath.value : '' + dPath: this.props.dPath || this.props.dPaths[0] }); } } diff --git a/common/components/WalletDecrypt/components/Mnemonic.tsx b/common/components/WalletDecrypt/components/Mnemonic.tsx index f255dbbb..0e3d30e1 100644 --- a/common/components/WalletDecrypt/components/Mnemonic.tsx +++ b/common/components/WalletDecrypt/components/Mnemonic.tsx @@ -27,7 +27,7 @@ interface State { formattedPhrase: string; pass: string; seed: string; - dPath: string; + dPath: DPath; } class MnemonicDecryptClass extends PureComponent { @@ -36,12 +36,12 @@ class MnemonicDecryptClass extends PureComponent { formattedPhrase: '', pass: '', seed: '', - dPath: this.props.dPath.value + dPath: this.props.dPath }; public componentWillReceiveProps(nextProps: Props) { if (this.props.dPath !== nextProps.dPath) { - this.setState({ dPath: nextProps.dPath.value }); + this.setState({ dPath: nextProps.dPath }); } } @@ -131,7 +131,7 @@ class MnemonicDecryptClass extends PureComponent { this.setState({ seed: '' }); }; - private handlePathChange = (dPath: string) => { + private handlePathChange = (dPath: DPath) => { this.setState({ dPath }); }; @@ -139,7 +139,7 @@ class MnemonicDecryptClass extends PureComponent { const { formattedPhrase, pass, dPath } = this.state; this.props.onUnlock({ - path: `${dPath}/${index}`, + path: `${dPath.value}/${index}`, pass, phrase: formattedPhrase, address diff --git a/common/components/WalletDecrypt/components/Trezor.tsx b/common/components/WalletDecrypt/components/Trezor.tsx index f2c09505..4b9e36f3 100644 --- a/common/components/WalletDecrypt/components/Trezor.tsx +++ b/common/components/WalletDecrypt/components/Trezor.tsx @@ -25,7 +25,7 @@ interface StateProps { interface State { publicKey: string; chainCode: string; - dPath: string; + dPath: DPath; error: string | null; isLoading: boolean; } @@ -36,14 +36,14 @@ class TrezorDecryptClass extends PureComponent { public state: State = { publicKey: '', chainCode: '', - dPath: this.props.dPath ? this.props.dPath.value : '', + dPath: this.props.dPath || this.props.dPaths[0], error: null, isLoading: false }; public componentWillReceiveProps(nextProps: Props) { - if (this.props.dPath !== nextProps.dPath) { - this.setState({ dPath: nextProps.dPath ? nextProps.dPath.value : '' }); + if (this.props.dPath !== nextProps.dPath && nextProps.dPath) { + this.setState({ dPath: nextProps.dPath }); } } @@ -98,19 +98,19 @@ class TrezorDecryptClass extends PureComponent { ); } - private handlePathChange = (dPath: string) => { + private handlePathChange = (dPath: DPath) => { this.setState({ dPath }); this.handleConnect(dPath); }; - private handleConnect = (dPath: string = this.state.dPath): void => { + private handleConnect = (dPath: DPath): void => { this.setState({ isLoading: true, error: null }); (TrezorConnect as any).getXPubKey( - dPath, + dPath.value, (res: any) => { if (res.success) { this.setState({ @@ -135,17 +135,19 @@ class TrezorDecryptClass extends PureComponent { }; private handleUnlock = (address: string, index: number) => { - this.props.onUnlock(new TrezorWallet(address, this.state.dPath, index)); + this.props.onUnlock(new TrezorWallet(address, this.state.dPath.value, index)); this.reset(); }; - private handleNullConnect = (): void => this.handleConnect(); + private handleNullConnect = (): void => { + this.handleConnect(this.state.dPath); + }; private reset() { this.setState({ publicKey: '', chainCode: '', - dPath: this.props.dPath ? this.props.dPath.value : '' + dPath: this.props.dPath || this.props.dPaths[0] }); } } diff --git a/common/selectors/config/wallet.ts b/common/selectors/config/wallet.ts index 02cddc08..f944dbdb 100644 --- a/common/selectors/config/wallet.ts +++ b/common/selectors/config/wallet.ts @@ -1,6 +1,6 @@ import { InsecureWalletName, SecureWalletName, WalletName, walletNames } from 'config'; import { EXTRA_PATHS } from 'config/dpaths'; -import sortedUniq from 'lodash/sortedUniq'; +import uniqBy from 'lodash/uniqBy'; import difference from 'lodash/difference'; import { StaticNetworkConfig, DPathFormats } from 'types/network'; import { AppState } from 'reducers'; @@ -22,7 +22,7 @@ export function getPaths(state: AppState, pathType: PathType): DPath[] { return networkPaths; }, []) .concat(EXTRA_PATHS); - return sortedUniq(paths); + return uniqBy(paths, p => `${p.label}${p.value}`); } export function getSingleDPath(state: AppState, format: DPathFormat): DPath | undefined { From 34ef8857da2deacbf0237664540306025a9de17c Mon Sep 17 00:00:00 2001 From: William O'Beirne Date: Mon, 30 Apr 2018 14:40:39 -0400 Subject: [PATCH 02/39] Trezor Help Link & Decrypt UI (#1695) --- common/components/WalletDecrypt/WalletDecrypt.scss | 4 ++++ common/components/WalletDecrypt/WalletDecrypt.tsx | 3 ++- common/components/WalletDecrypt/components/Trezor.tsx | 2 +- 3 files changed, 7 insertions(+), 2 deletions(-) diff --git a/common/components/WalletDecrypt/WalletDecrypt.scss b/common/components/WalletDecrypt/WalletDecrypt.scss index 186a6c92..6d9bb4f9 100644 --- a/common/components/WalletDecrypt/WalletDecrypt.scss +++ b/common/components/WalletDecrypt/WalletDecrypt.scss @@ -59,6 +59,10 @@ $speed: 500ms; text-align: center; padding-bottom: $space; + @media (max-width: $screen-md) { + padding-bottom: $space * 2; + } + &-back { @include reset-button; position: absolute; diff --git a/common/components/WalletDecrypt/WalletDecrypt.tsx b/common/components/WalletDecrypt/WalletDecrypt.tsx index 7fbe7489..b9515f8c 100644 --- a/common/components/WalletDecrypt/WalletDecrypt.tsx +++ b/common/components/WalletDecrypt/WalletDecrypt.tsx @@ -166,7 +166,8 @@ const WalletDecrypt = withRouter( component: TrezorDecrypt, initialParams: {}, unlock: this.props.setWallet, - helpLink: 'https://doc.satoshilabs.com/trezor-apps/mew.html' + helpLink: + 'https://support.mycrypto.com/accessing-your-wallet/how-to-use-your-trezor-with-mycrypto.html' }, [SecureWalletName.PARITY_SIGNER]: { lid: 'X_PARITYSIGNER', diff --git a/common/components/WalletDecrypt/components/Trezor.tsx b/common/components/WalletDecrypt/components/Trezor.tsx index 4b9e36f3..6bc27a81 100644 --- a/common/components/WalletDecrypt/components/Trezor.tsx +++ b/common/components/WalletDecrypt/components/Trezor.tsx @@ -79,7 +79,7 @@ class TrezorDecryptClass extends PureComponent {
{error || '-'}
- + How to use TREZOR with MyCrypto
From 9d1a218a0e4e53eaf41d0e86c04f46b80f9653a7 Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Mon, 30 Apr 2018 21:07:59 -0400 Subject: [PATCH 03/39] Fix Estimate Gas (#1698) * Properly pad hex encoded data * Remove to address for gas estimation if its zero value --- common/libs/nodes/rpc/utils.ts | 9 +++++++-- common/sagas/transaction/network/gas.ts | 10 ++++++++++ 2 files changed, 17 insertions(+), 2 deletions(-) diff --git a/common/libs/nodes/rpc/utils.ts b/common/libs/nodes/rpc/utils.ts index 1c7cd164..eccb8dd3 100644 --- a/common/libs/nodes/rpc/utils.ts +++ b/common/libs/nodes/rpc/utils.ts @@ -1,7 +1,7 @@ // Ref: https://github.com/ethereum/wiki/wiki/JSON-RPC import BN from 'bn.js'; -import { toBuffer, addHexPrefix, bufferToHex } from 'ethereumjs-util'; +import { toBuffer, addHexPrefix, bufferToHex, stripHexPrefix, padToEven } from 'ethereumjs-util'; import trimStart from 'lodash/trimStart'; // When encoding QUANTITIES (integers, numbers): encode as hex, prefix with "0x", the most compact representation (slight exception: zero should be represented as "0x0"). @@ -12,5 +12,10 @@ export function hexEncodeQuantity(value: BN | Buffer): string { // When encoding UNFORMATTED DATA (byte arrays, account addresses, hashes, bytecode arrays): encode as hex, prefix with "0x", two hex digits per byte. export function hexEncodeData(value: string | Buffer): string { - return bufferToHex(toBuffer(value)); + // convert the value to a buffer + // convert the value to a hex prefixed hex string + // strip the hex prefix + // pad the data to even (two hex digits per byte) + // add the hex prefix back in + return addHexPrefix(padToEven(stripHexPrefix(bufferToHex(toBuffer(value))))); } diff --git a/common/sagas/transaction/network/gas.ts b/common/sagas/transaction/network/gas.ts index d915d452..422a4bf4 100644 --- a/common/sagas/transaction/network/gas.ts +++ b/common/sagas/transaction/network/gas.ts @@ -78,6 +78,15 @@ export function* shouldEstimateGas(): SagaIterator { transaction ); + // gas estimation calls with + // '0x' as an address (contract creation) + // fail, so instead we set it as undefined + // interestingly, the transaction itself as '0x' as the + // to address works fine. + if (rest.to === '0x') { + rest.to = undefined as any; + } + yield put(estimateGasRequested(rest)); } } @@ -101,6 +110,7 @@ export function* estimateGas(): SagaIterator { try { const from: string = yield apply(walletInst, walletInst.getAddressString); const txObj = { ...payload, from }; + const { gasLimit } = yield race({ gasLimit: apply(node, node.estimateGas, [txObj]), timeout: call(delay, 10000) From d348e3c112e744f4420688229855950bc27e9760 Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Tue, 1 May 2018 21:45:02 -0400 Subject: [PATCH 04/39] Fix retry scan if you have no tokens (#1708) * Fix retry scan if you have no tokens. * Swap out updateTokenBalances for retryTokenBalances --- common/sagas/wallet/wallet.ts | 23 +++++++++++++++++++---- 1 file changed, 19 insertions(+), 4 deletions(-) diff --git a/common/sagas/wallet/wallet.ts b/common/sagas/wallet/wallet.ts index 7272fad8..9bb49841 100644 --- a/common/sagas/wallet/wallet.ts +++ b/common/sagas/wallet/wallet.ts @@ -76,6 +76,18 @@ export function* updateAccountBalance(): SagaIterator { } } +export function* retryTokenBalances(): SagaIterator { + const tokens: MergedToken[] = yield select(getWalletConfigTokens); + if (tokens && tokens.length) { + yield call(updateTokenBalances); + } else { + const wallet: null | IWallet = yield select(getWalletInst); + if (wallet) { + yield call(scanWalletForTokens, wallet); + } + } +} + export function* updateTokenBalances(): SagaIterator { try { const isOffline = yield select(getOffline); @@ -126,14 +138,17 @@ export function* updateTokenBalance(action: SetTokenBalancePendingAction): SagaI } } -export function* scanWalletForTokens(action: ScanWalletForTokensAction): SagaIterator { +export function* handleScanWalletAction(action: ScanWalletForTokensAction): SagaIterator { + yield call(scanWalletForTokens, action.payload); +} + +export function* scanWalletForTokens(wallet: IWallet): SagaIterator { try { const isOffline = yield select(getOffline); if (isOffline) { return; } - const wallet = action.payload; const tokens: MergedToken[] = yield select(getTokens); yield put(setTokenBalancesPending()); @@ -274,11 +289,11 @@ export default function* walletSaga(): SagaIterator { takeEvery(TypeKeys.WALLET_UNLOCK_KEYSTORE, unlockKeystore), takeEvery(TypeKeys.WALLET_UNLOCK_MNEMONIC, unlockMnemonic), takeEvery(TypeKeys.WALLET_SET, handleNewWallet), - takeEvery(TypeKeys.WALLET_SCAN_WALLET_FOR_TOKENS, scanWalletForTokens), + takeEvery(TypeKeys.WALLET_SCAN_WALLET_FOR_TOKENS, handleScanWalletAction), takeEvery(TypeKeys.WALLET_SET_WALLET_TOKENS, handleSetWalletTokens), takeEvery(TypeKeys.WALLET_SET_TOKEN_BALANCE_PENDING, updateTokenBalance), takeEvery(TypeKeys.WALLET_REFRESH_ACCOUNT_BALANCE, updateAccountBalance), - takeEvery(TypeKeys.WALLET_REFRESH_TOKEN_BALANCES, updateTokenBalances), + takeEvery(TypeKeys.WALLET_REFRESH_TOKEN_BALANCES, retryTokenBalances), // Foreign actions takeEvery(ConfigTypeKeys.CONFIG_TOGGLE_OFFLINE, updateBalances), takeEvery(CustomTokenTypeKeys.CUSTOM_TOKEN_ADD, handleCustomTokenAdd) From dc8edc91b7d4260458844914f4ae842d6789b524 Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Tue, 1 May 2018 21:45:22 -0400 Subject: [PATCH 05/39] Remove checksumming of keystore address (#1697) --- common/libs/web-workers/workers/generateKeystore.worker.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/common/libs/web-workers/workers/generateKeystore.worker.ts b/common/libs/web-workers/workers/generateKeystore.worker.ts index f8737351..b65da3c2 100644 --- a/common/libs/web-workers/workers/generateKeystore.worker.ts +++ b/common/libs/web-workers/workers/generateKeystore.worker.ts @@ -1,5 +1,4 @@ import { generate } from 'ethereumjs-wallet'; -import { toChecksumAddress } from 'ethereumjs-util'; const worker: Worker = self as any; @@ -14,6 +13,6 @@ worker.onmessage = (event: MessageEvent) => { const filename = wallet.getV3Filename(); const privateKey = wallet.getPrivateKeyString(); const keystore = wallet.toV3(info.password, { n: info.N_FACTOR }); - keystore.address = toChecksumAddress(keystore.address); + worker.postMessage({ keystore, filename, privateKey }); }; From 347b2f0684955f2cb835a805b86b36f3859659be Mon Sep 17 00:00:00 2001 From: Daniel Ternyak Date: Tue, 1 May 2018 21:13:38 -0500 Subject: [PATCH 06/39] Use proxy.mycryptoapi.com for rates (#1710) --- common/api/rates.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/common/api/rates.ts b/common/api/rates.ts index 21d43344..5c7dcfff 100644 --- a/common/api/rates.ts +++ b/common/api/rates.ts @@ -47,11 +47,11 @@ export const rateSymbols: IRateSymbols = { // TODO - internationalize const ERROR_MESSAGE = 'Could not fetch rate data.'; -const CCApi = 'https://min-api.cryptocompare.com'; +const CCApi = 'https://proxy.mycryptoapi.com/cc'; const CCRates = (symbols: string[]) => { const tsyms = rateSymbols.symbols.all.concat(symbols as any).join(','); - return `${CCApi}/data/price?fsym=ETH&tsyms=${tsyms}`; + return `${CCApi}/price?fsym=ETH&tsyms=${tsyms}`; }; export interface CCResponse { From 9e844b71f98b87618d61e79b801e3e9e431f5bc5 Mon Sep 17 00:00:00 2001 From: Daniel Ternyak Date: Tue, 1 May 2018 21:15:27 -0500 Subject: [PATCH 07/39] Update Release Candidate Version (#1709) --- common/config/data.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/common/config/data.tsx b/common/config/data.tsx index 12719e0b..7637c0d9 100644 --- a/common/config/data.tsx +++ b/common/config/data.tsx @@ -10,7 +10,7 @@ export const discordURL = 'https://discord.gg/VSaTXEA'; // Displays in the footer export const VERSION_RAW = packageJson.version; -export const VERSION = `${VERSION_RAW} (Release Candidate)`; +export const VERSION = `${VERSION_RAW} (Release Candidate 2)`; export const N_FACTOR = 8192; // Bricks the app once this date has been exceeded. Remember to update these 2 From a8740e0fc2747da7245d1bf644a99a8923af130d Mon Sep 17 00:00:00 2001 From: William O'Beirne Date: Tue, 8 May 2018 23:06:35 -0400 Subject: [PATCH 08/39] Show version in Electron (#1730) * Add app version to both support us page and help menu * Typescriptery --- common/containers/Tabs/SupportPage/index.tsx | 9 ++++++++- electron-app/main/menu.ts | 10 ++++++++-- 2 files changed, 16 insertions(+), 3 deletions(-) diff --git a/common/containers/Tabs/SupportPage/index.tsx b/common/containers/Tabs/SupportPage/index.tsx index 1fc08374..f9e7c97e 100644 --- a/common/containers/Tabs/SupportPage/index.tsx +++ b/common/containers/Tabs/SupportPage/index.tsx @@ -2,7 +2,13 @@ import React from 'react'; import translate from 'translations'; import TabSection from 'containers/TabSection'; import logo from 'assets/images/logo-mycrypto-transparent.svg'; -import { donationAddressMap, socialMediaLinks, productLinks, affiliateLinks } from 'config'; +import { + donationAddressMap, + socialMediaLinks, + productLinks, + affiliateLinks, + VERSION +} from 'config'; import DisclaimerModal from 'components/DisclaimerModal'; import { NewTabLink } from 'components/ui'; import './index.scss'; @@ -47,6 +53,7 @@ export default class SupportPage extends React.Component<{}, State> { +
v{VERSION}
diff --git a/electron-app/main/menu.ts b/electron-app/main/menu.ts index d2655135..43035994 100644 --- a/electron-app/main/menu.ts +++ b/electron-app/main/menu.ts @@ -1,5 +1,6 @@ import { MenuItemConstructorOptions, shell } from 'electron'; import { APP_TITLE, REPOSITORY } from '../constants'; +import packageJson from '../../package.json'; const MENU: MenuItemConstructorOptions[] = [ { @@ -32,9 +33,14 @@ const MENU: MenuItemConstructorOptions[] = [ } ]; -const HELP_MENU = { +const HELP_MENU: MenuItemConstructorOptions = { role: 'help', submenu: [ + { + label: `v${packageJson.version}`, + enabled: false + }, + { type: 'separator' }, { label: 'Help / FAQ', click() { @@ -68,7 +74,7 @@ if (process.platform === 'darwin') { MENU.push({ ...HELP_MENU, submenu: [ - ...HELP_MENU.submenu, + ...(HELP_MENU.submenu as MenuItemConstructorOptions[]), { label: 'Speech', submenu: [{ role: 'startspeaking' }, { role: 'stopspeaking' }] From 2ed8caf0783171feff35ec4647cc8d2639f9c7b5 Mon Sep 17 00:00:00 2001 From: Kazunori Seki Date: Wed, 9 May 2018 12:47:12 +0900 Subject: [PATCH 09/39] Japanese V4 full translation (#1655) * Japanese V4 full translation PIck all the labels from en.js and sorted in the same order. * merge conflicts --- common/translations/lang/ja.json | 951 ++++++++++++++----------------- 1 file changed, 423 insertions(+), 528 deletions(-) diff --git a/common/translations/lang/ja.json b/common/translations/lang/ja.json index 215919c7..02606ab5 100644 --- a/common/translations/lang/ja.json +++ b/common/translations/lang/ja.json @@ -1,116 +1,52 @@ { "code": "ja", "data": { - "HELP_2A_TITLE": "お財布の保管やバックアップの方法は? ", - "X_CANCELREPLACETX": "処理を中断、あるいは置換", - "X_CANCELTX": "処理を中断", - "X_PASSWORDDESC": "このパスワードで秘密鍵を*暗号化*します。新しい鍵を作るための元種(seed)ではありません。**このパスワードと(暗号化された)秘密鍵の二つを使って、お財布を解錠します**", - "X_READMORE": "もっと読む", - "X_REPLACETX": "処理を置き換える", - "X_TRANSHASH": "処理ハッシュ", - "X_TXFEE": "処理料", "X_TXHASH": "処理ハッシュ", + "X_PASSWORDDESC": "このパスワードで秘密鍵を*暗号化*します。新しい鍵を作るための元種(seed)ではありません。**このパスワードと(暗号化された)秘密鍵の二つを使って、お財布を解錠します**", "NAV_CHECKTXSTATUS": "処理状況を確認", "NAV_TXSTATUS": "処理状況", - "TX_DETAILS": "処理内容詳細", - "TX_SUMMARY": "もし数日経ってから処理状況を確認した場合でなければ、大量の処理発生時(ICO期間など)には、数時間待たされることがあります。本ツールは、そのような状況において処理待ちのものを探し出し取り消す、あるいは新しくする機能を提供します。**これは一般的な操作ではありませんが、処理プールが満杯の場合にのみ有効です。 [このツールに関しては、こちらを参考にしてください。](https://support.mycrypto.com/topics/how-can-i-check-on-the-status-of-my-transaction-can-i-cancel-override-overwrite-replace-or-do-anything-once-a-transaction-has-been-sent)**", "TX_NOTFOUND": "対象の処理が見つかりません", "TX_NOTFOUND_1": "この処理は、現在接続中の処理プールの中にありませんでした。", "TX_NOTFOUND_2": "もし今、処理を送出した直後であれば、15秒待ってから「処理状況を確認」ボタンを再度押してください。", "TX_NOTFOUND_3": "別の処理プールで発掘待ちになっているかもしれません。", "TX_NOTFOUND_4": "右上の下展開メニューから、別のノードを選択してください。 (例: `ETH (Etherscan.io)` or `ETH (Infura.io)` or `ETH (MyCrypto)`) から選んで再度確認する。", - "TX_FOUNDINPENDING": "待機中の処理が見つかりました。", - "TX_FOUNDINPENDING_1": "あなたの処理が、現在接続中のノードの処理待ちプールの中で見つかりました。", - "TX_FOUNDINPENDING_2": "現在待機中です(発掘待ち)。", - "TX_FOUNDINPENDING_3": "この処理を取り消す、あるいは置き換えることができます。下記のお財布をアンロックしてください。 ", - "TX_FOUNDONCHAIN": "処理が見つかりました", - "TX_FOUNDONCHAIN_1": "あなたの待機中の処理は発掘されてブロックチェーンに載りました。", - "TX_FOUNDONCHAIN_2": "**もし赤い `( ! )`, `BAD INSTRUCTION` あるいは `OUT OF GAS` のエラーメッセージを見つけたら**, これは、処理送出に失敗したということです。処理の取り消しや置き換えはできません。代わりに、新しい処理を送出してください。 \"Out of Gas\" エラーの場合には, ガスリミットをはじめに指定した値の倍にしてください。", - "TX_FOUNDONCHAIN_3": "**何もエラーメッセージが返ってこなければ、あなたの処理は正しく送出されています。** ETHあるいはトークンは、送ろうとしたあて先の場所にあります。 もし、ETHやトークンが他のお財布や交換所のお財布に見つからず、処理を開始してから24時間以上経っていたら、 [そのサービスに連絡](https://support.mycrypto.com/topics/i-have-a-question-but-its-not-about-myetherwallet-dot-com-what-do-i-do)してください。状況を確認してもらうために、自分の処理のリンクをうまく知らせてください。", - "GEN_HELP_1": "この", - "GEN_HELP_2": "で自分の口座にアクセスしてください。", - "GEN_HELP_3": "自分のデバイスそのものが、自分のお財布です。", - "GEN_HELP_4": "ガイドとFAQ", - "GEN_HELP_5": "お財布の作り方", - "GEN_HELP_6": "ここから始める", - "GEN_HELP_7": "安全な所で保管してください · バックアップを作成してください · 他の誰にも教えないでください · 絶対になくさないでください · 無くした時には回復する方法はありません。", - "GEN_HELP_8": "ファイルをダウンロードしませんでしたか?", - "GEN_HELP_9": "Google Chromeをお使いください", - "GEN_HELP_10": "右クリックしてファイルを保存。ファイル名: ", - "GEN_HELP_11": "このファイルは自分のコンピューターで開かないでください", - "GEN_HELP_12": "MyCryptoの上でこれを使って自分のお財布をアンロックしてください(Mist, Geth, Parityや他のお財布クライアントも可)", - "GEN_HELP_13": "自分のキーストアファイルのバックアップ作成方法", - "GEN_HELP_14": "これらの方式の違いは?", - "GEN_HELP_15": "自分の資金の紛失や盗難を防止する。", - "GEN_HELP_16": "これらの方式の違いは?", - "GEN_HELP_17": "なぜこれらをすべきか?", - "GEN_HELP_18": "2番目のバックアップ作成のため", - "GEN_HELP_19": "パスワードを忘れた場合には", - "GEN_HELP_20": "隔離された保管場所", - "GET_CONFBUTTON": "理解できました。続けます。", - "GEN_LABEL_5": "自分の秘密鍵を保存する。", - "GEN_UNLOCK": "自分のアドレスを確認するために、お財布を解錠する", - "GAS_PRICE_DESC": "ガス価格は、ガスの一単位にかかる料金のことです。 「処理料金 = ガス価格 * ガスリミット」かつ、自分の処理をブロックに配置するためにマイナーに支払われます。ガス価格が高いほど処理は早く行われますが、料金は高くなります。デファルトは 「21 GWEI」です。", - "GAS_LIMIT_DESC": "ガスリミットは、自分の処理にかかる料金の額です。「使用料金」 = ガス価格 * ガスリミット」で、自分の処理をブロックに配置するための料金に支払われます。 この数字を増やしても、自分の処理が早く発掘されることはありません。ETHの送出 = 「21000」。トークンの送出 = ~「200000」。", - "NONCE_DESC": "そのnonceは、指定のアドレスから送出される処理の数です。処理が正しい順番で重複しないように確実にするためのものです。", - "TXFEE_DESC": "その処理料金は自分の処理をブロックに配置するためにマイナーに支払われます。「ガスリミット」*「ガス価格」です。 [GWEI -> ETHの変換はここです。](https://www.mycrypto.com/helpers.html)", - "NAV_ADDWALLET": "お財布の追加 ", - "NAV_BULKGENERATE": "バルク作成 ", - "NAV_CONTACT": "連絡する ", + "TX_NOTFOUND_5": "TXハッシュが正しくコピーされていることを確認してください。", + "GEN_LABEL_5": "自分の秘密鍵を保存", "NAV_CONTRACTS": "契約 ", "NAV_DEPLOYCONTRACT": "契約を展開 ", "NAV_ENS": "ENS", - "NAV_GENERATEWALLET_ALT": "New Wallet ", + "NAV_VIEW": "表示と送出", "NAV_GENERATEWALLET": "お財布の作成 ", "NAV_HELP": "ヘルプ ", - "NAV_INTERACTCONTRACT": "契約を操作 ", - "NAV_MULTISIG": "多重署名 ", - "NAV_MYWALLETS": "自分のお財布 ", - "NAV_OFFLINE": "オフライン送出 ", + "NAV_SIGN": "メッセージの署名と確認", + "NAV_BROADCAST": "処理の一斉同報通信", "NAV_SENDETHER": "Ether/トークンの送出 ", - "NAV_SENDTOKENS": "トークン送出 ", + "NAV_REQUESTPAYMENT": "支払い請求", "NAV_SIGNMSG": "メッセージ署名 ", - "NAV_SWAP": "通貨の両替 ", - "NAV_VIEWWALLET": "お財布情報を見る ", - "NAV_YOURWALLETS": "自分のお財布 ", + "NAV_SWAP": "両替 ", + "NAV_VIEWWALLET": "お財布情報を表示 ", + "NAV_RECENT_TX": "直近の処理", + "NAV_SUPPORT_US": "ご支援ください", "X_ACCESS": "アクセス ", - "X_ADDESSDESC": "Your Address can also be known as you `Account #` or your `Public Key`. It is what you share with people so they can send you Ether or Tokens. Find the colorful address icon. Make sure it matches your paper wallet & whenever you enter your address somewhere. これは自分のアカウント番号と公開鍵になります。ETHを送信するために必要な情報です。アイコンは自分のアドレスを識別するものです。 ", "X_ADDRESS": "自分のアドレス ", - "X_CANCEL": "取り消す ", - "X_CIPHER": "Cipher Browser", - "X_CSV": "CSV ファイル (未暗号化) ", - "X_DOWNLOAD": "ダウンロード ", - "X_JSON": "JSON ファイル (未暗号化) ", - "X_JSONDESC": "これはパスワードが不要な暗号化されていないJSONフォーマットの秘密鍵です。この暗号化されていないJSONフォーマットの秘密鍵を使えば、誰でもパスワードを使わずに自分のお財布とEtherにアクセスできます。 ", + "X_CIPHER": "暗号化ブラウザ", "X_KEYSTORE": "Keystore ファイル (UTC / JSON · 推奨 · 暗号化) ", "X_KEYSTORE2": "Keystore ファイル ", "X_KEYSTOREDESC": "この Keystore / JSON ファイルは、後で容易にインポートするため、Mistで使われているフォーマットと一致させる必要があります。ダウンロードしてバックアップを取ることをおすすめします。 ", - "X_LEDGER": "Ledger ", "X_METAMASK": "Metamask / Mist ", + "X_MIST": "Mist", + "X_WEB3": "Web3", "X_MNEMONIC": "ニーモニック文節 ", - "X_PARITYPHRASE": "パリティ文節 ", - "X_PASSWORD": "パスワード ", "X_PRINT": "お財布紙情報を印刷 ", - "X_PRINTDESC": "助言: プリンターが接続されていなくても、「印刷」をクリックしてPDFで保存できます。 ", - "X_PRINTSHORT": "印刷 ", + "X_PRINTDESC": "プロの助言 プリンターが接続されていなくても、「印刷」をクリックしてPDFで保存できます。 ", "X_PRIVKEY": "秘密鍵(未暗号化) ", "X_PRIVKEY2": "秘密鍵 ", "X_PRIVKEYDESC": "これはパスワードが不要な暗号化されていない秘密鍵です。この暗号化されていない秘密鍵を使えば、誰でもパスワードなしで自分のお財布を使用できます。従って、暗号化された秘密鍵の利用をおすすめします。 ", "X_SAVE": "保存する ", - "X_TREZOR": "TREZOR ", - "X_TXT": "TXT ファイル (未暗号化) ", - "X_WALLET": "お財布 ", - "MYCRYPTO_WARNING_1": "お財布を操作したり新しいお財布を作成する前に、必ずURLを確認してください。詐欺サイトに御注意ください! ", + "X_TRY_AGAIN": "もう一度お試しください", + "X_CUSTOM": "カスタム ", "CX_WARNING_1": "必ずここに保管するすべてのお財布の「外部バックアップ」を作成してください。このChrome Extensionでは、再導入などで、データが保存されずの消失することが頻繁に起こります。このエクステンションは容易な操作を提供するのみで、「バックアップ」は行いません。 ", - "MYCRYPTO_TAGLINE": "オープンソース JavaScript クライアントサイド Etherお財布 ", - "CX_TAGLINE": "オープンソース JavaScript クライアントサイド Etherお財布 Chrome Extension ", - "FOOTER_1": "イサーリアムお財布の作成とトランザクション実行のためのオープンソース、javascript、 クライアントサイドツール。 ", - "FOOTER_1B": "制作 ", - "FOOTER_2": "投げ銭に感謝致します!: ", - "FOOTER_3": "クライアントサイドお財布制作 ", - "FOOTER_4": "免責事項 ", - "SIDEBAR_ACCOUNTINFO": "アカウント情報 ", - "SIDEBAR_ACCOUNTADDR": "アカウントアドレス: ", + "SIDEBAR_ACCOUNTADDR": "アカウントアドレス ", "SIDEBAR_ACCOUNTBAL": "アカウント残高 ", "SIDEBAR_TOKENBAL": "トークン残高 ", "SIDEBAR_EQUIV": "等価 ", @@ -118,135 +54,33 @@ "SIDEBAR_DONATION": "MyCryptoは、プライバシーとセキュリティのための無料のオープンソースサービスです。 寄付が増えることによって、新機能やフィードバックの反映を行い、よりユーザーの皆様のご希望に沿った制作の時間を増やす事が可能になります。私たちは、たった二人で世界を変えようとしています。お手伝いいただけますか? ", "SIDEBAR_DONATE": "寄付する ", "SIDEBAR_THANKS": "感謝します!!! ", - "SIDEBAR_DISPLAYONTREZOR": "Display address on TREZOR", - "SIDEBAR_DISPLAYONLEDGER": "Display address on Ledger", - "CX_ERROR_1": "お財布が保存されていません。[\"Add Wallet\"](/cx-wallet.html#add-wallet)をクリックして追加してください。 ", - "CX_QUICKSEND": "簡易送信 ", + "SIDEBAR_DISPLAY_ADDR": "$wallet でアドレス表示 ", "DECRYPT_ACCESS": "どの方法でお財布を操作しますか? ", - "DECRYPT_TITLE": "秘密鍵形式の選択: ", - "DECRYPT_SELECT": "お財布の選択 ", - "ADD_LABEL_1": "ご希望の操作方法をお選びください ", - "ADD_RADIO_1": "お財布の新規作成 ", + "X_LEDGER": "Ledger ", + "ADD_LEDGER_SCAN": "Ledger Walletに接続 ", + "ADD_METAMASK": "MetaMaskに接続 ", + "X_TREZOR": "TREZOR ", + "ADD_TREZOR_SCAN": "TREZORに接続する ", + "X_PARITYSIGNER": "Parity Signer ", + "ADD_PARITY_DESC": "Parity Signerモバイルアプリ経由で接続て署名 ", + "ADD_PARITY_1": "処理が取り消されました ", + "ADD_PARITY_2": "アプリをダウンロード ", + "ADD_PARITY_3": "スキャン", + "ADD_PARITY_4": "詳細は以下を参照 [Parity Wiki]($wiki_link).", + "ADD_PARITY_ERROR_DISABLED": "Parity Signerをアンロックするためには、カメラを有効にしなければなりません", + "ADD_PARITY_ERROR_NO_CAM": "Parity Signerをアンロックするためには、カメラが必要です", + "ADD_PARITY_ERROR_UNKNOWN": "カメラに接続できませんでした。再表示するか別のブラウザーをお使いください", "ADD_RADIO_2": "お財布ファイルの選択 (Keystore / JSON) ", "ADD_RADIO_2_ALT": "お財布ファイルの選択: ", "ADD_RADIO_2_SHORT": "お財布ファイルを選択 ", - "ADD_RADIO_3": "秘密鍵をペースト/タイプ ", - "ADD_RADIO_4": "監視するアカウントを追加 ", - "ADD_RADIO_5": "ニーモニックを上書き/タイプ ", - "ADD_RADIO_5_PATH": "HD derivation pathを選択 ", - "ADD_RADIO_5_WOTREZOR": "(Jaxx, Metamask, Exodus, imToken)", - "ADD_RADIO_5_WITHTREZOR": "(Jaxx, Metamask, Exodus, imToken, TREZOR)", - "ADD_RADIO_5_PATHALTERNATIVE": "(Ledger)", - "ADD_RADIO_5_PATHTREZOR": "(TREZOR)", - "ADD_RADIO_5_PATHCUSTOM": "カスタム", - "ADD_LABEL_2": "ニックネームの作成: ", "ADD_LABEL_3": "お財布が暗号化されています。パスワードを入力してください: ", - "ADD_LABEL_4": "監視するアカウントを追加 ", - "ADD_WARNING_1": "任意のアカウントを監視する目的で、秘密鍵をアップロードせずにお財布タブに追加することできます。これによってお財布の操作や、ETHERの移動が可能になるわけではありません。 ", "ADD_LABEL_5": "アドレスを入力 ", "ADD_LABEL_6": "お財布をアンロック: ", "ADD_LABEL_6_SHORT": "アンロック ", "ADD_LABEL_7": "アカウント追加 ", - "ADD_LABEL_8": "Password (optional): ", - "MNEM_1": "操作したいアドレスを選択してください。 ", - "MNEM_2": "HDニーモニックの一つの文節で、複数のお財布やアドレスが操作可能です。操作したいアドレスを選択してください。 ", - "MNEM_MORE": "さらにアドレスを表示 ", - "MNEM_PREV": "前のアドレス表示 ", - "ADD_LEDGER_1": "自分の Ledger Wallet を接続する ", - "ADD_LEDGER_2": "イサリアムアプリケーション(あるいはコントラクトアプリケーション)を開く 。 ", - "ADD_LEDGER_3": "設定中で、ブラウザサポートが有効にされていることを確認してください。 ", - "ADD_LEDGER_4": "設定にブラウザサポートが見つからなければ、[Firmware >1.2]を確認してください。(https://www.ledgerwallet.com/apps/manager) ", - "ADD_LEDGER_0A": "セキュアコネクション(SSL)で再度MyCryptoを開いてください。 ", - "ADD_LEDGER_0B": "MyCryptoを再度「Chrome」(https://www.google.com/chrome/browser/desktop/) あるいは [Opera](https://www.opera.com/)で開いてください。 ", - "ADD_LEDGER_SCAN": "Ledger Wallet に接続 ", - "ADD_METAMASK": "Connect to MetaMask ", - "ADD_TREZOR_SCAN": "TREZORに接続する ", - "ADD_TREZOR_SELECT": "これはTREZORのシードです ", - "X_DIGITALBITBOX": "Digital Bitbox ", - "ADD_DIGITALBITBOX_0A": "セキュアコネクション(SSL)で再度MyCryptoを開いてください。 ", - "ADD_DIGITALBITBOX_0B": "MyCryptoを再度「Chrome」(https://www.google.com/chrome/browser/desktop/) あるいは [Opera](https://www.opera.com/)で開いてください。 ", - "ADD_DIGITALBITBOX_SCAN": "DigitalBitboxに接続する ", - "GEN_DESC": "複数のお財布の作成をこのタブで行う事ができます: ", - "GEN_LABEL_1": "強固なパスワードを入力(9文字以上) ", - "GEN_PLACEHOLDER_1": "必ず保存してください! ", - "GEN_SUCCESSMSG": "成功!お財布が作成されました。 ", - "GEN_LABEL_2": "Keystore/JSON あるいは秘密鍵を保存してください。パスワードを絶対に忘れないようにしてください。 ", - "GEN_LABEL_3": "アドレスを保存してください。 ", - "GEN_LABEL_4": "必要であれば、お財布紙情報、あるいはQRコードを印刷してください。 ", - "GEN_ARIA_1": "", - "GEN_ARIA_2": "", - "BULK_LABEL_1": "作成するお財布の数 ", - "BULK_LABEL_2": "お財布を複数を作成する ", - "BULK_SUCCESSMSG": "成功!お財布が作成されました。 ", - "SEND_ADDR": "宛先アドレス: ", - "SEND_AMOUNT": "送出数量: ", - "SEND_AMOUNT_SHORT": "数量 ", - "SEND_GAS": "ガス ", - "SEND_TRANSFERTOTAL": "残高をすべて送出する ", - "SEND_GENERATE": "トランザクションを生成 ", - "SEND_RAW": "未加工トランザクション ", - "SEND_SIGNED": "署名済みトランザクション ", - "SEND_TRANS": "トランザクションの送出 ", - "SENDMODAL_TITLE": "警告! ", - "SEND_CUSTOM": "カスタムトークンを追加 ", - "SENDMODAL_CONTENT_1": "送出準備が ", - "SENDMODAL_CONTENT_2": "について、アドレス ", - "SENDMODAL_CONTENT_3": "に対し完了。本当に送出してよろしいですか? ", - "SENDMODAL_CONTENT_4": "ノート:最も起こりやすいエラーの原因は、送出のためのガス不足で、その場合には、ETHを追加する必要があります。ガスはETHで支払われます。 ", - "SENDMODAL_NO": "いいえ、中断します。 ", - "SENDMODAL_YES": "はい、確かです。処理を実行します。 ", - "TOKEN_ADDR": "アドレス: ", - "TOKEN_SYMBOL": "トークンシンボル: ", - "TOKEN_DEC": "ケタ数: ", - "TOKEN_SHOW": "全てのトークンを表示 ", - "TOKEN_HIDE": "トークンを隠す ", - "TRANS_DESC": "トークンを送出するには「トークン送出」のタブを選択してください。 ", - "TRANS_WARNING": "コントラクト上で、「ETHのみ」あるいは「ETCのみ」の送出を行う場合、受け入れサービスによって、これらのトランザクションに問題を生ずる場合があります。以下参照。 ", - "TRANS_ADVANCED": "+Advanced: ガスあるいはデータを追加してください ", - "TRANS_DATA": "データ: ", - "TRANS_GAS": "ガスリミット: ", - "TRANS_SENDINFO": "21000ガスを使用する標準トランザクションは、0.000441 ETHを消費します。迅速な処理を行うために、若干最小量よりも多めの0.000000021 ETHのガスを使用します。当サービスでは、トランザクション料金は徴収いたしません。 ", - "TRANSMODAL_TITLE": "「ETHのみ」と「ETCのみ」トランザクション ", - "TRANSMODAL_CONTENT_0": "異なるトランザクションと異なるサービスの注釈: ", - "TRANSMODAL_CONTENT_1": "**ETH(標準トランザクション): ** これは、アドレス間移動のデファルトのトランザクションを生成します。デフォルトガス値は21000です。このメソッドで送出されたETHは、高い可能性でETCチェーンでリプレイされます。 ", - "TRANSMODAL_CONTENT_2": "**ETHのみ: ** [Timon Rappのリプレイコントラクト(VB推奨)](https://blog.ethereum.org/2016/07/26/onward_from_the_hard_fork/) を使用して、**ETH**チェーンのみに送出します。 ", - "TRANSMODAL_CONTENT_3": "**ETCのみ: ** [Timon Rappのリプレイコントラクト(VB推奨)](https://blog.ethereum.org/2016/07/26/onward_from_the_hard_fork/) を使用して、**ETC**チェーンのみに送出します。 ", - "TRANSMODAL_CONTENT_4": "**Coinbase & ShapeShift: ** スタンダードトランザクションのみで送出します。どちらかのみのコントラクトで送出する場合には、サポートスタッフに連絡して、手動で残高に追加したり払い戻しをする必要があります。[Shapeshiftの「スプリット」ツールも使用可能です。(https://split.shapeshift.io/) ", - "TRANSMODAL_CONTENT_5": "**Kraken & Poloniex:** 問題は確認されていません。どれでもお使いください。 ", - "TRANSMODAL_YES": "理解しました。 ", - "TRANSMODAL_NO": "わかりません。おしえてください。 ", - "OFFLINE_TITLE": "オフライントランザクションを作成し送出 ", - "OFFLINE_DESC": "オフライントランザクションの作成は、3ステップで行う事ができます。ステップ1と3はオンラインのコンピューター上で行い、ステップ2は、オフライン、あるいは物理的にネットワークと切断されたコンピューターを用います。これにより、秘密鍵が、インターネットに接続したデバイスと接触する事を避ける事ができます。 ", - "OFFLLINE_STEP1_TITLE": "ステップ1: 情報生成 (オンラインコンピューター) ", - "OFFLINE_STEP1_BUTTON": "情報生成 ", - "OFFLINE_STEP1_LABEL_1": "送出元アドレス: ", - "OFFLINE_STEP1_LABEL_2": "ノート:これは、送出元アドレスであって、送出先アドレスではありません。操作元アカウントからは「Nonce」が生成されます。切断されたコンピューターを使用する場合に、このアドレスはコールドストレージのアカウントのものになります。 ", - "OFFLINE_STEP2_TITLE": "ステップ2: 情報生成 (オフラインコンピューター) ", - "OFFLINE_STEP2_LABEL_1": "送出先アドレス: ", - "OFFLINE_STEP2_LABEL_2": "送出する値/総量 ", - "OFFLINE_STEP2_LABEL_3": "ガス価格 ", - "OFFLINE_STEP2_LABEL_3B": "これは、ステップ1でオンラインコンピューターに表示されたものです。 ", - "OFFLINE_STEP2_LABEL_4": "ガスリミット ", - "OFFLINE_STEP2_LABEL_4B": "デフォルトガスリミット値は21000です。契約や付加データーを送出する場合には、これらの値は 異なるものにする必要があります。使用されなかったガスは全て返却されます。 ", - "OFFLINE_STEP2_LABEL_5": "Nonce ", - "OFFLINE_STEP2_LABEL_5B": "これらは、ステップ1でオンラインコンピューターに表示されたものです。 ", - "OFFLINE_STEP2_LABEL_6": "データ ", - "OFFLINE_STEP2_LABEL_6B": "これは、任意の付加データです。契約に対してトランザクションを送出する際などでよく使われます。 ", - "OFFLINE_STEP2_LABEL_7": "秘密鍵/JSONの入力/選択 ", - "OFFLINE_STEP3_TITLE": "ステップ3: トランザクションの送出/公開(オンラインコンピューター) ", - "OFFLINE_STEP3_LABEL_1": "ステップ2で署名されたトランザクションをここにペーストして「トランザクションの送出」ボタンをクリックする。 ", - "DEP_GENERATE": "バイトコードを生成する ", - "DEP_GENERATED": "生成されたバイトコード ", - "DEP_SIGNTX": "トランザクションに署名 ", - "DEP_INTERFACE": "生成されたインターフェース ", - "CONTRACT_TITLE": "契約アドレス ", - "CONTRACT_TITLE_2": "既存の契約を選択 ", - "CONTRACT_JSON": "ABI / JSON インターフェース ", - "CONTRACT_INTERACT_TITLE": "契約の読込や書込 ", - "CONTRACT_INTERACT_CTA": "機能を一つ選ぶ ", - "CONTRACT_BYTECODE": "バイトコード ", - "CONTRACT_READ": "読み取り ", - "CONTRACT_WRITE": "書き込み ", + "ADD_LABEL_8": "パスワード (随意) ", + "ADD_WEB3DESC": "自分のブラウザーあるいは拡張から接続して署名してください", + "ADD_HARDWAREDESC": "Connect & sign via your hardware wallet", "MYWAL_NICK": "お財布ニックネーム ", "MYWAL_ADDRESS": "お財布アドレス ", "MYWAL_BAL": "残高 ", @@ -260,14 +94,108 @@ "MYWAL_EDIT_2": "お財布を編集: ", "MYWAL_NAME": "お財布名 ", "MYWAL_CONTENT_1": "警告! お財布を消去しようとしています: ", - "MYWAL_CONTENT_2": "**秘密鍵と鍵保存ファイル及びパスワード**が保管されているか確認してください。 ", "MYWAL_CONTENT_3": "MyCrypto CXでこのお財布を使用するためには、秘密鍵あるいはJSONとパスワードを手動で追加する必要があります。 ", + "GEN_DESC": "複数のお財布の作成をこのタブで行う事ができます: ", + "GEN_LABEL_1": "パスワードを入力", + "GEN_PLACEHOLDER_1": "必ず保存してください! ", + "GEN_SUCCESSMSG": "成功!お財布が作成されました。 ", + "GEN_LABEL_2": "自分の保存キーファイルを保存", + "GEN_LABEL_3": "自分のアドレスを保存", + "GEN_LABEL_4": "ワレット情報、あるいはQRコードを印刷", + "GEN_ARIA_1": "強力なパスワード(少なくとも9文字以上)を入力してください。", + "GEN_ARIA_2": "パスワードを見えるようにする", + "BULK_LABEL_1": "作成するお財布の数 ", + "BULK_LABEL_2": "複数のお財布を作成する ", + "BULK_SUCCESSMSG": "成功!お財布が作成されました。 ", + "SEND_ADDR": "宛先アドレス: ", + "SEND_ADDR_SHORT": "To ", + "SEND_AMOUNT": "送出数量: ", + "SEND_AMOUNT_SHORT": "数量 ", + "SEND_CUSTOM": "カスタムトークンを追加 ", + "SCAN_TOKENS": "トークンをスキャン", + "SCAN_TOKENS_FAIL": "トークン価の取得に失敗", + "SCAN_TOKENS_FAIL_NO_TOKENS": "トークンが見つかりませんでした", + "SCAN_TOKENS_OFFLINE": "オフラインでトークンの値は取得できません", + "SCHEDULING_TOGGLE": "後で送出 ", + "SCHEDULING_TITLE": "予約されたトランザクションの設定", + "SCHEDULING_DESCRIPTION": "トランザクションの送出を時間をずらして後で行うことができます。将来のネットワークの混雑など予測できない状況のため、設定した時間の枠内で必ずしも送出されるものではありません。", + "SCHEDULE_BLOCK": "ブロックナンバー ", + "SCHEDULE_BLOCK_PLACEHOLDER": "ブロックナンバーを入力 ", + "SCHEDULE_DEPOSIT": "デポジット(証拠金)が必要です。(オプション)", + "SCHEDULE_DEPOSIT_TOOLTIP": "実行のための排他的な枠をタイムノードに確保するために、ETHの必要量をデポジットする必要があります。", + "SCHEDULE_TIMESTAMP": "日付及び時間 ", + "SCHEDULE_TIMEZONE": "タイムゾーン ", + "SCHEDULE_TIMEBOUNTY": "タイムバウンティー ", + "SCHEDULE_TIMEBOUNTY_PLACEHOLDER": "タイムバウンティーを入力 ", + "SCHEDULE_TIMEBOUNTY_TOOLTIP": "実行のためタイムノードに提供するETHの値です。このタイムバウンティーが大きければ大きいほど、トランザクションが実行されやすくなります。 ", + "SCHEDULE_CHECK": "Chronosでチェック ", + "SCHEDULE_SCHEDULE": "トランザクションの予約 ", + "SCHEDULE_TYPE_TIME": "分 ", + "SCHEDULE_TYPE_BLOCK": "ブロック ", + "SCHEDULE_GAS_PRICE": "先(未来)のガス価格", + "SCHEDULE_GAS_LIMIT": "先(未来)のガスリミット", + "SCHEDULE_WINDOW_SIZE": "枠 ", + "SCHEDULE_WINDOW_SIZE_TOOLTIP_TIME": "トランザクションが実行され得る分刻みの時間領域。 ", + "SCHEDULE_WINDOW_SIZE_TOOLTIP_BLOCK": "トランザクションが実行され得るブロックナンバーの領域。 ", + "SEND_GAS": "ガス ", + "SEND_TRANSFERTOTAL": "残高をすべて送出する ", + "SEND_GENERATE": "トランザクションを生成 ", + "SEND_RAW": "未加工トランザクション ", + "SEND_SIGNED": "署名済みトランザクション ", + "SEND_TRANS": "トランザクションの送出 ", + "SENDMODAL_TITLE": "警告! ", + "SENDMODAL_CONTENT_1": "送出準備が ", + "SENDMODAL_CONTENT_2": "について、アドレス ", + "SENDMODAL_CONTENT_3": "に対し完了。本当に送出してよろしいですか? ", + "SENDMODAL_CONTENT_4": "ノート:最も起こりやすいエラーの原因は、送出のためのガス不足で、その場合には、ETHを追加する必要があります。ガスはETHで支払われます。 ", + "TOKEN_ADDR": "アドレス: ", + "TOKEN_SYMBOL": "トークンシンボル: ", + "TOKEN_DEC": "ケタ数: ", + "TOKEN_SHOW": "全てのトークンを表示 ", + "TOKEN_HIDE": "トークンを隠す ", + "TRANS_ADVANCED": "+Advanced ガスあるいはデータを追加してください ", + "TRANS_SIMPLE": "シンプル", + "TRANS_DATA": "データ: ", + "TRANS_GAS": "ガスリミット: ", + "TRANS_SENDINFO": "21000ガスを使用する標準トランザクションは、0.000441 ETHを消費します。迅速な処理を行うために、若干最小量よりも多めの0.000000021 ETHのガスを使用します。当サービスでは、トランザクション料金は徴収いたしません。 ", + "OFFLINE_TITLE": "オフライントランザクションを生成・送出する ", + "OFFLINE_DESC": "オフライントランザクションの作成は、3ステップで行う事ができます。ステップ1と3はオンラインのコンピューター上で行い、ステップ2は、オフライン、あるいは物理的にネットワークと切断されたコンピューターを用います。これにより、秘密鍵が、インターネットに接続したデバイスと接触する事を避ける事ができます。 ", + "OFFLLINE_STEP1_TITLE": "ステップ1:情報生成(オンラインコンピューター) ", + "OFFLINE_STEP1_BUTTON": "情報生成 ", + "OFFLINE_STEP1_LABEL_1": "送出元アドレス: ", + "OFFLINE_STEP1_LABEL_2": "ノート:これは、送出元アドレスであって、送出先アドレスではありません。操作元アカウントからは「Nonce」が生成されます。切断されたコンピューターを使用する場合に、このアドレスはコールドストレージのアカウントのものになります。 ", + "OFFLINE_STEP2_TITLE": "ステップ2 情報生成 (オフラインコンピューター) ", + "OFFLINE_STEP2_LABEL_1": "送出先アドレス: ", + "OFFLINE_STEP2_LABEL_2": "送出する値/総量 ", + "OFFLINE_STEP2_LABEL_3": "ガス価格 ", + "OFFLINE_STEP2_LABEL_3B": "これは、ステップ1でオンラインコンピューターに表示されたものです。 ", + "OFFLINE_STEP2_LABEL_4": "ガスリミット ", + "OFFLINE_STEP2_LABEL_4B": "デフォルトガスリミット値は21000です。契約や付加データーを送出する場合には、これらの値は 異なるものにする必要があります。使用されなかったガスは全て返却されます。 ", + "OFFLINE_STEP2_LABEL_5": "Nonce ", + "OFFLINE_STEP2_LABEL_5B": "これらは、ステップ1でオンラインコンピューターに表示されたものです。 ", + "OFFLINE_STEP2_LABEL_6": "データ ", + "OFFLINE_STEP2_LABEL_6B": "これは、任意の付加データです。契約に対してトランザクションを送出する際などでよく使われます。 ", + "OFFLINE_STEP2_LABEL_7": "秘密鍵/JSONの入力/選択 ", + "OFFLINE_STEP3_TITLE": "ステップ3:トランザクションを送出・公開する (Online Computer) ", + "CONTRACT_TITLE": "契約アドレス ", + "CONTRACT_TITLE_2": "既存の契約を選択 ", + "CONTRACT_JSON": "ABI / JSON インターフェース ", + "CONTRACT_INTERACT_TITLE": "契約の読込や書込 ", + "CONTRACT_INTERACT_CTA": "機能を一つ選ぶ ", + "CONTRACT_BYTECODE": "バイトコード ", + "CONTRACT_READ": "読み取り ", + "CONTRACT_WRITE": "書き込み ", + "DEP_GENERATE": "バイトコードを生成 ", + "DEP_GENERATED": "生成されたバイトコード ", + "DEP_SIGNTX": "トランザクションに署名 ", + "DEP_INTERFACE": "生成されたインターフェース ", "NODE_TITLE": "カスタムノードをセットアップ", "NODE_SUBTITLE": "ローカルノードに接続するには...", - "NODE_WARNING": "MyCrypto.com 経由で接続するためには、HTTPSノードが必要です。[MyCryptoをレポジトリからダウンロードして、手元で走らせ、](https://download.mycrypto.com/)いずれかのノードに接続することもできます。 あるいは、無料のSSL証明書を[LetsEncrypt](https://letsencrypt.org/)から入手してください", + "NODE_WARNING": "MyCrypto.com 経由で接続するためには、HTTPSノードが必要です。[MyCryptoをレポジトリからダウンロードして、手元で走らせ、](https//github.com/MyCryptoHQ/MyCrypto/releases/latest)いずれかのノードに接続することもできます。 あるいは、無料のSSL証明書を[LetsEncrypt](https//letsencrypt.org/)から入手してください", "NODE_NAME": "ノード名", "NODE_PORT": "ノードポート", "NODE_CTA": "保存してカスタムノードを使用", + "NODE_ADD": "カスタムノードの追加", "SWAP_RATES": "現在のレート ", "SWAP_INIT_1": "これから両替で ", "SWAP_INIT_2": " を ", @@ -279,12 +207,13 @@ "SWAP_REC_ADD": "受け取りアドレス ", "SWAP_START_CTA": "交換開始 ", "SWAP_REF_NUM": "参照番号 ", - "SWAP_TIME": "送出するまでにあと、 ", - "SWAP_ELAPSED": "Time elapsed since sent ", + "SWAP_TIME": "送出するまでにあと、", + "SWAP_ELAPSED": "送出してからの経過時間 ", "SWAP_PROGRESS_1": "注文を開始しました ", "SWAP_PROGRESS_2": "到着待機中 ", "SWAP_PROGRESS_3": "ETH受け取り完了 ", "SWAP_PROGRESS_4": "送出中 $destination_id ", + "SWAP_PROGRESS_CONFIRMATIONS": "$number_confirmations 件の確認を待機中...", "SWAP_PROGRESS_5": "注文完了 ", "SWAP_ORDER_CTA": "お送りいただきたいのは ", "SWAP_UNLOCK": "このページから直接ETHあるいはトークンを送出するためには、お財布を解錠してください。", @@ -295,11 +224,12 @@ "MSG_INFO1": "この署名が別の日付で再度使われないようにするために、現時点の日付を入れてください。 ", "MSG_INFO2": "他人に使われないようにするため、あなたのニックネームとそれが使われるところを入れてください。 ", "MSG_INFO3": "異なった目的で使用されないようにするために、利用目的を入れてください。 ", - "VIEWWALLET_SUBTITLE": "異なったバージョンの秘密鍵をダウンロードしたり、お財布紙情報を再印刷することができます。[import your account into Geth/Mist](https://ethereum.stackexchange.com/questions/465/how-to-import-a-plain-private-key-into-geth/)する時に必要です。残高をチェックするためには、[etherscan.io](https://etherscan.io/)のようなブロックチェーンエクスプローラーサービスを使う事をおすすめします。 ", + "VIEWWALLET_SUBTITLE": "異なったバージョンの秘密鍵をダウンロードしたり、お財布紙情報を再印刷することができます。[import your account into Geth/Mist](https//ethereum.stackexchange.com/questions/465/how-to-import-a-plain-private-key-into-geth/)する時に必要です。残高をチェックするためには、[etherscan.io](https//etherscan.io/)のようなブロックチェーンエクスプローラーサービスを使う事をおすすめします。 ", "VIEWWALLET_SUBTITLE_SHORT": "異なったバージョンの秘密鍵をダウンロードしたり、お財布紙情報を再印刷することができます。 ", "VIEWWALLET_SUCCESSMSG": "成功しました! お財布の詳細は以下の通りです。 ", "VIEWWALLET_SHOWPRIVKEY": "(show)", "VIEWWALLET_HIDEPRIVKEY": "(hide)", + "CX_QUICKSEND": "簡易送信 ", "ERROR_0": "正しい値を入力してください。 ", "ERROR_1": "パスワードは少なくとも9文字が必要です。強固なパスワードであることをお確かめください。 ", "ERROR_2": "申し訳ございませんが、このタイプのお財布ファイルは認識できません。 ", @@ -327,338 +257,303 @@ "ERROR_24": "正しいポートを入力してください ", "ERROR_25": "正しい chain ID を入力してください ", "ERROR_26": "正しい ABI を入力してください ", - "ERROR_27": "最小値: 0.01. 最大値: ", - "ERROR_28": "お財布を操作するためには**Keystore/JSONとパスワードか秘密鍵が必要** 保存してから、外部バックアップしてください!ここで保存しないとお財布が使用できなくなります。詳細はヘルプページを参照してください。(https://www.mycrypto.com/#help) ", + "ERROR_27": "最小値 0.01. 最大値 ", "ERROR_29": "正しいユーザーとパスワードを入力してください ", "ERROR_30": "正しい ENS名を入力してください ", "ERROR_31": "無効な秘密フレーズです ", - "ERROR_32": "ノードに接続できませんでした。Refresh your page, try a different node (upper right corner), check your firewall settings. If custom node, check your configs.", - "ERROR_33": "The wallet you have unlocked does not match the owner's address. ", - "ERROR_34": "The name you are attempting to reveal does not match the name you have entered. ", - "ERROR_35": "Input address is not checksummed. More info", - "ERROR_36": "Enter valid TX hash", - "ERROR_37": "Enter valid hex string (0-9, a-f)", + "ERROR_32": "ノードに接続できませんでした。ページを再表示、(右上にある)別のノードを試す、あるいはファイアーウォール設定を確認してください。カスタムノードの場合は、設定を確認してください。", + "ERROR_33": "今アンロックしたワレットのアドレスが、所有者のアドレスと一致していません。", + "ERROR_34": "開札しようとした名前が入力されたものと一致していません。", + "ERROR_36": "正しいトランザクションハッシュを入力してください。", + "ERROR_37": "正しいHEX文字列(0-9, a-f)を入力してください。", "SUCCESS_1": "有効なアドレス ", "SUCCESS_2": "お財布は正常に暗号解除されました。 ", - "SUCCESS_3": "トランザクションはブロックチェイン上に展開されています。そのトランザクションを表示し、ガス不足や契約実行エラーがないことを確認しするためにクリックしてください。 TX Hash: ", + "SUCCESS_3": "トランザクションはブロックチェイン上に展開されています。そのトランザクションを表示し、ガス不足契約実行エラーがないことを確認しするためにクリックしてください。 TX Hash ", "SUCCESS_4": "お財布が追加されました: ", "SUCCESS_5": "選択されました: ", "SUCCESS_6": "接続完了しました ", "SUCCESS_7": "メッセージの署名が確認されました", "WARN_SEND_LINK": "自分のアドレス、リンク、ガス、データ、あるいはトランザクションタイプ(送出モード)が指定されたリンクでここに表示されています。 送出前に修正可能です。もう一度行うために、お財布を解錠してください。 ", - "PARITY_ALREADYIMPORTED": "同じハッシュのトランザクションがすでにインポートされています。", - "PARITY_OLD": "トランザクション nonceが小さすぎます。増やしてみてください。", - "PARITY_TOOCHEAPTOREPLACE": "トランザクションフィー不足です。同じ nonce のトランザクションが別のキューにあります。fee または nonce を増やしてみてください。", - "PARITY_LIMITREACHED": "キューにあるトランザクションの数が多すぎます。上限を超えたため、対象のトランザクションは除外されています。その fee を増やしてみてください。", - "PARITY_INSUFFICIENTGASPRICE": "トランザクションフィー不足です。 ノードの最小 fee を満たしていません。 (minimal: {}, got: {}). fee を増やしてください。", - "PARITY_INSUFFICIENTBALANCE": "ファンドが足りません。 トランザクション送出元のファンドが不足しています。 必要量 {} で 現在: {}  です。", - "PARITY_GASLIMITEXCEEDED": "トランザクションコストがガスリミットを超過しました。 リミット: {}, 現在: {}. ガス供給量を減らして見てください。", - "PARITY_INVALIDGASLIMIT": "ガス供給量が制限を超過しています。", - "GETH_INVALIDSENDER": "送出元が無効です ", - "GETH_NONCE": "Nonce が足りません ", - "GETH_CHEAP": "ガス価格が低すぎます ", "GETH_BALANCE": "残高不足 ", - "GETH_NONEXISTENTACCOUNT": "アカウントが存在しない、あるいはその残高不足です ", - "GETH_INSUFFICIENTFUNDS": "ガス*価格+数量に足りません ", - "GETH_INTRINSICGAS": "基本のガス不足です ", "GETH_GASLIMIT": "ブロックガスリミットを越えています ", - "GETH_NEGATIVEVALUE": "負の値です ", - "TRANSLATE_VERSION": "0.3 ", - "TRANSLATOR_DESC": "日本語開発者に投げ銭: ", - "TRANSLATORNAME_1": "[sekisanchi](https://www.mycrypto.com/?gaslimit=21000&to=0xf991119Eea62Eee1a6fdaA7f621e91A42f325FcE&value=1.0#send-transaction) ", - "TRANSLATORADDR_1": "0xf991119Eea62Eee1a6fdaA7f621e91A42f325FcE ", - "TRANSLATORNAME_2": "", - "TRANSLATORADDR_2": "", - "TRANSLATORNAME_3": "", - "TRANSLATORADDR_3": "", - "TRANSLATORNAME_4": "", - "TRANSLATORADDR_4": "", - "TRANSLATORNAME_5": "", - "TRANSLATORADDR_5": "", - "HELP_WARNING": "If you created a wallet -or- downloaded the repo before **Dec. 31st, 2015**, please check your wallets & download a new version of the repo. Click for details. ", - "HELP_DESC": "Do you see something missing? Have another question? [Get in touch with us](mailto:support@mycrypto.com), and we will not only answer your question, we will update this page to be more useful to people in the future! ", - "HELP_REMIND_TITLE": "Some reminders ", - "HELP_REMIND_DESC_1": "**Ethereum, MyCrypto.com & MyCrypto CX, and some of the underlying Javascript libraries we use are under active development.** While we have thoroughly tested & tens of thousands of wallets have been successfully created by people all over the globe, there is always the remote possibility that something unexpected happens that causes your ETH to be lost. Please do not invest more than you are willing to lose, and please be careful. If something were to happen, we are sorry, but **we are not responsible for the lost Ether**. ", - "HELP_REMIND_DESC_2": "MyCrypto.com & MyCrypto CX are not \"web wallets\". You do not create an account or give us your Ether to hold onto. All data never leaves your computer/your browser. We make it easy for you to create, save, and access your information and interact with the blockchain. ", - "HELP_REMIND_DESC_3": "If you do not save your private key & password, there is no way to recover access to your wallet or the funds it holds. Back them up in multiple physical locations – not just on your computer! ", - "HELP_0_TITLE": "0) I'm new. What do I do? ", - "HELP_0_DESC_1": "MyCrypto gives you the ability to generate new wallets so you can store your Ether yourself, not on an exchange. This process happens entirely on your computer, not our servers. Therefore, when you generate a new wallet, **you are responsible for safely backing it up**. ", - "HELP_0_DESC_2": "Create a new wallet. ", - "HELP_0_DESC_3": "Back the wallet up. ", - "HELP_0_DESC_4": "Verify you have access to this new wallet and have correctly saved all necessary information. ", - "HELP_0_DESC_5": "Transfer Ether to this new wallet. ", - "HELP_1_TITLE": "1) How do I create a new wallet? ", - "HELP_1_DESC_1": "Go to the \"Generate Wallet\" page. ", - "HELP_1_DESC_2": "Go to the \"Add Wallet\" page & select \"Generate New Wallet\" ", - "HELP_1_DESC_3": "Enter a strong password. If you think you may forget it, save it somewhere safe. You will need this password to send transactions. ", - "HELP_1_DESC_4": "Click \"GENERATE\". ", - "HELP_1_DESC_5": "Your wallet has now been generated. ", - "HELP_2A_DESC_1": "You should always back up your wallet externally and in multiple physical locations - like on a USB drive and/or a piece of paper. ", - "HELP_2A_DESC_2": "Save the address. You can keep it to yourself or share it with others. That way, others can transfer ether to you. ", - "HELP_2A_DESC_3": "Save versions of the private key. Do not share it with anyone else. Your private key is necessary when you want to access your Ether to send it! There are 3 types of private keys: ", - "HELP_2A_DESC_4": "Place your address, versions of the private key, and the PDF version of your paper wallet in a folder. Save this on your computer and a USB drive. ", - "HELP_2A_DESC_5": "Print the wallet if you have a printer. Otherwise, write down your private key and address on a piece of paper. Store this as a secure location, separate from your computer and the USB drive. ", - "HELP_2A_DESC_6": "Keep in mind, you must prevent loss of the keys and password due to loss or failure of you hard drive failure, or USB drive, or piece of paper. You also must keep in mind physical loss / damage of an entire area (think fire or flood). ", - "HELP_2B_TITLE": "2b) How do I safely / offline / cold storage with MyCrypto? ", - "HELP_2B_DESC_1": "Go to [https://download.mycrypto.com/](https://download.mycrypto.com/). ", - "HELP_2B_DESC_2": "Click on `etherwallet-vX.X.X.X.zip`. ", - "HELP_2B_DESC_3": "Move zip to an airgapped computer. ", - "HELP_2B_DESC_4": "Unzip it and double-click `index.html`. ", - "HELP_2B_DESC_5": "Generate a wallet with a strong password. ", - "HELP_2B_DESC_6": "Save the address. Save versions of the private key. Save the password if you might not remember it forever. ", - "HELP_2B_DESC_7": "Store these papers / USBs in multiple physically separate locations. ", - "HELP_2B_DESC_8": "Go to the \"View Wallet Info\" page and type in your private key / password to ensure they are correct and access your wallet. Check that the address you wrote down is the same. ", - "HELP_3_TITLE": "3) How do I verify I have access to my new wallet? ", - "HELP_3_DESC_1": "**Before you send any Ether to your new wallet**, you should ensure you have access to it. ", - "HELP_3_DESC_2": "Navigate to the \"View Wallet Info\" page. ", - "HELP_3_DESC_3": "Navigate to the MyCrypto.com \"View Wallet Info\" page. ", - "HELP_3_DESC_4": "Select your wallet file -or- your private key and unlock your wallet. ", - "HELP_3_DESC_5": "If the wallet is encrypted, a text box will automatically appear. Enter the password. ", - "HELP_3_DESC_6": "Click the \"Unlock Wallet\" button. ", - "HELP_3_DESC_7": "Your wallet information should show up. Find your account address, next to a colorful, circular icon. This icon visually represents your address. Be certain that the address is the address you have saved to your text document and is on your paper wallet. ", - "HELP_3_DESC_8": "If you are planning on holding a large amount of ether, we recommend that send a small amount of ether from new wallet before depositing a large amount. Send 0.001 ether to your new wallet, access that wallet, send that 0.001 ether to another address, and ensure everything works smoothly. ", - "HELP_4_TITLE": "4) How do I send Ether from one wallet to another? ", - "HELP_4_DESC_1": "If you plan to move a large amount of ether, you should test sending a small amount to your wallet first to ensure everything goes as planned. ", - "HELP_4_DESC_2": "Navigate to the \"Ether送出 トークン送出\" page. ", - "HELP_4_DESC_3": "Select your wallet file -or- your private key and unlock your wallet. ", - "HELP_4_DESC_4": "If the wallet is encrypted, a text box will automatically appear. Enter the password. ", - "HELP_4_DESC_5": "Click the \"Unlock Wallet\" button. ", - "HELP_4_DESC_6": "Enter the address you would like to send to in the \"To Address:\" field. ", - "HELP_4_DESC_7": "Enter the amount you would like to send. You can also click the \"Send Entire Balance\" link if you would like the transfer the entire balance. ", - "HELP_4_DESC_9": "Click \"Generate Transaction\". ", - "HELP_4_DESC_10": "A couple more fields will appear. This is your browser generating the transaction. ", - "HELP_4_DESC_11": "Click the blue \"Send Transaction\" button below that. ", - "HELP_4_DESC_12": "A pop-up will appear. Verify that the amount and the address you are sending to are correct. Then click \"Yes, I am sure! Make transaction.\" button. ", - "HELP_4_DESC_13": "The transaction will be submitted. The TX Hash will display. You can click that TX Hash to see it on the blockchain. ", - "HELP_4CX_TITLE": "4) How do I send Ether using MyCrypto CX? ", - "HELP_4CX_DESC_1": "First, you need to add a wallet. Once you have done that, you have 2 options: the \"QuickSend\" functionality from the Chrome Extension icon or the \"Ether送出 トークン送出\" page. ", - "HELP_4CX_DESC_2": "QuickSend: ", - "HELP_4CX_DESC_3": "Click the Chrome Extension Icon. ", - "HELP_4CX_DESC_4": "Click the \"QuickSend\" button. ", - "HELP_4CX_DESC_5": "Select the wallet you wish to send from. ", - "HELP_4CX_DESC_6": "Enter the address you would like to send to in the \"To Address:\" field. ", - "HELP_4CX_DESC_7": "Enter the amount you would like to send. You can also click the \"Send Entire Balance\" link if you would like the transfer the entire balance. ", - "HELP_4CX_DESC_8": "Click \"Send Transaction\". ", - "HELP_4CX_DESC_9": "Verify the address and the amount you are sending is correct. ", - "HELP_4CX_DESC_10": "Enter the password for that wallet. ", - "HELP_4CX_DESC_11": "Click \"Send Transaction.\" ", - "HELP_4CX_DESC_12": "Using \"Ether送出 トークン送出\" Page ", - "HELP_5_TITLE": "5) How do I run MyCrypto.com offline/locally? ", - "HELP_5_DESC_1": "You can run MyCrypto.com on your computer instead of from the GitHub servers. You can generate a wallet completely offline and send transactions from the \"Offline Transaction\" page. ", - "HELP_5_DESC_7": "MyCrypto.com is now running entirely on your computer. ", - "HELP_5_DESC_8": "In case you are not familiar, you need to keep the entire folder in order to run the website, not just `index.html`. Don't touch or move anything around in the folder. If you are storing a backup of the MyCrypto repo for the future, we recommend just storing the ZIP so you can be sure the folder contents stay intact. ", - "HELP_5_DESC_9": "As we are constantly updating MyCrypto.com, we recommend you periodically update your saved version of the repo. ", - "HELP_5CX_TITLE": "5) How can I install this extension from the repo instead of the Chrome Store? ", - "HELP_5CX_DESC_2": "Click on `chrome-extension-vX.X.X.X.zip` and unzip it. ", - "HELP_5CX_DESC_3": "Go to Google Chrome and find you settings (in the menu in the upper right). ", - "HELP_5CX_DESC_4": "Click \"Extensions\" on the left. ", - "HELP_5CX_DESC_5": "Check the \"Developer Mode\" button at the top of that page. ", - "HELP_5CX_DESC_6": "Click the \"Load unpacked extension...\" button. ", - "HELP_5CX_DESC_7": "Navigate to the now-unzipped folder that you downloaded earlier. Click \"select\". ", - "HELP_5CX_DESC_8": "The extension should now show up in your extensions and in your Chrome Extension bar. ", - "HELP_7_TITLE": "7) How do I send Tokens & add custom tokens? ", - "HELP_7_DESC_0": "[Ethplorer.io](https://ethplorer.io/) is a great way to explore tokens and find the decimals of a token. ", - "HELP_7_DESC_1": "Navigate to the \"Ether送出 トークン送出\" page. ", - "HELP_7_DESC_2": "Unlock your wallet. ", - "HELP_7_DESC_3": "Enter the address you would like to send to in the \"To Address:\" field. ", - "HELP_7_DESC_4": "Enter the amount you would like to send. ", - "HELP_7_DESC_5": "Select which token you would like to send. ", - "HELP_7_DESC_6": "If you do not see the token listed: ", - "HELP_7_DESC_7": "Click \"Custom\". ", - "HELP_7_DESC_8": "Enter the address, name, and decimals of the token. These are provided by the developers of the token and are also needed when you \"Add a Watch Token\" to Mist. ", - "HELP_7_DESC_9": "Click \"Save\". ", - "HELP_7_DESC_10": "You can now send that token as well as see it's balance in the sidebar. ", - "HELP_7_DESC_11": "Click \"Generate Transaction\". ", - "HELP_7_DESC_12": "A couple more fields will appear. This is your browser generating the transaction. ", - "HELP_7_DESC_13": "Click the blue \"Send Transaction\" button below that. ", - "HELP_7_DESC_14": "A pop-up will appear. Verify that the amount and the address you are sending to are correct. Then click \"Yes, I am sure! Make transaction.\" button. ", - "HELP_7_DESC_15": "The transaction will be submitted. The TX Hash will display. You can click that TX Hash to see it on the blockchain. ", - "HELP_8_TITLE": "8) What happens if your site goes down? ", - "HELP_8_DESC_1": "MyCrypto is not a web wallet. You don't have a login and nothing ever gets saved to our servers. It is simply an interface that allows you interact with the blockchain. ", - "HELP_8_DESC_2": "If MyCrypto.com goes down, you would have to find another way (like geth or Ethereum Wallet / Mist) to do what we are doing. But you wouldn't have to \"get\" your Ether out of MyCrypto because it's not in MyCrypto. It's in whatever wallet your generated via our site. ", - "HELP_8_DESC_3": "You can import your unencrypted private key and your Geth/Mist Format (encrypted) files directly into geth / Ethereum Wallet / Mist very easily now. See question #12 below. ", - "HELP_8_DESC_4": "In addition, the likelihood of us taking MyCrypto down is slim to none. It costs us almost nothing to maintain as we aren't storing any information. If we do take the domain down, it still is, and always will be, publicly available at [https://github.com/MyCryptoHQ/MyCrypto](https://github.com/MyCryptoHQ/MyCrypto/tree/gh-pages). You can download the ZIP there and run it locally. ", - "HELP_8CX_TITLE": "8) What happens if MyCrypto CX disappears? ", - "HELP_8CX_DESC_1": "First, all data is saved on your computer, not our servers. I know it can be confusing, but when you look at the Chrome Extension, you are NOT looking at stuff saved on our servers somewhere - it's all saved on your own computer. ", - "HELP_8CX_DESC_2": "That said, it is **very important** that you back up all your information for any new wallets generated with MyCrypto CX. That way if anything happens to MyCrypto CX or your computer, you still have all the information necessary to access your Ether. See the #2a for how to back up your wallets. ", - "HELP_8CX_DESC_3": "If for some reason MyCrypto CX disappears from the Chrome Store, you can find the source on Github and load it manually. See #5 above. ", - "HELP_9_TITLE": "9) Is the \"Ether送出 トークン送出\" page offline? ", - "HELP_9_DESC_1": "No. It needs the internet in order to get the current gas price, nonce of your account, and broadcast the transaction (aka \"send\"). However, it only sends the signed transaction. Your private key safely stays with you. We also now provide an \"Offline Transaction\" page so that you can ensure your private keys are on an offline/airgapped computer at all times. ", - "HELP_10_TITLE": "10) How do I make an offline transaction? ", - "HELP_10_DESC_1": "Navigate to the \"Offline Transaction\" page via your online computer. ", - "HELP_10_DESC_2": "Enter the \"From Address\". Please note, this is the address you are sending FROM, not TO. This generates the nonce and gas price. ", - "HELP_10_DESC_3": "Move to your offline computer. Enter the \"TO ADDRESS\" and the \"AMOUNT\" you wish to send. ", - "HELP_10_DESC_4": "Enter the \"GAS PRICE\" as it was displayed to you on your online computer in step #1. ", - "HELP_10_DESC_5": "Enter the \"NONCE\" as it was displayed to you on your online computer in step #1. ", - "HELP_10_DESC_6": "The \"GAS LIMIT\" has a default value of 21000. This will cover a standard transaction. If you are sending to a contract or are including additional data with your transaction, you will need to increase the gas limit. Any excess gas will be returned to you. ", - "HELP_10_DESC_7": "If you wish, enter some data. If you enter data, you will need to include more than the 21000 default gas limit. All data is in HEX format. ", - "HELP_10_DESC_8": "Select your wallet file -or- your private key and unlock your wallet. ", - "HELP_10_DESC_9": "Press the \"GENERATE SIGNED TRANSACTION\" button. ", - "HELP_10_DESC_10": "The data field below this button will populate with your signed transaction. Copy this and move it back to your online computer. ", - "HELP_10_DESC_11": "On your online computer, paste the signed transaction into the text field in step #3 and click send. This will broadcast your transaction. ", - "HELP_12_TITLE": "12) How do I import a wallet created with MyCrypto into geth / Ethereum Wallet / Mist? ", - "HELP_12_DESC_1": "Using an Geth/Mist JSON file from MyCrypto v2+.... ", - "HELP_12_DESC_2": "Go to the \"View Wallet Info\" page. ", - "HELP_12_DESC_3": "Unlock your wallet using your **encrypted** private key or JSON file. ", - "HELP_12_DESC_4": "Go to the \"My Wallets\" page. ", - "HELP_12_DESC_5": "Select the wallet you want to import into Mist, click the \"View\" icon, enter your password, and access your wallet. ", - "HELP_12_DESC_6": "Find the \"Download JSON file - Geth/Mist Format (encrypted)\" section. Press the \"Download\" button below that. You now have your keystore file. ", - "HELP_12_DESC_7": "Open the Ethereum Wallet application. ", - "HELP_12_DESC_8": "In the menu bar, go \"Accounts\" -> \"Backup\" -> \"Accounts\" ", - "HELP_12_DESC_9": "This will open your keystore folder. Copy the file you just downloaded (`UTC--2016-04-14......../`) into that keystore folder. ", - "HELP_12_DESC_10": "Your account should show up immediately under \"Accounts.\" ", - "HELP_12_DESC_11": "Using your unencrypted private key... ", - "HELP_12_DESC_12": "If you do not already have your unencrypted private key, navigate to the \"View Wallet Details\" page. ", - "HELP_12_DESC_13": "Select your wallet file -or- enter/paste your private key to unlock your wallet. ", - "HELP_12_DESC_14": "Copy Your Private Key (unencrypted). ", - "HELP_12_DESC_15": "If you are on a Mac: ", - "HELP_12_DESC_15B": "If you are on a PC: ", - "HELP_12_DESC_16": "Open Text Edit and paste this private key. ", - "HELP_12_DESC_17": "Go to the menu bar and click \"Format\" -> \"Make Plain Text\". ", - "HELP_12_DESC_18": "Save this file to your `desktop/` as `nothing_special_delete_me.txt`. Make sure it says \"UTF-8\" and \"If no extension is provided use .txt\" in the save dialog. ", - "HELP_12_DESC_19": "Open terminal and run the following command: `geth account import ~/Desktop/nothing_special_delete_me.txt` ", - "HELP_12_DESC_20": "This will prompt you to make a new password. This is the password you will use in geth / Ethereum Wallet / Mist whenever you send a transaction, so don't forget it. ", - "HELP_12_DESC_21": "After successful import, delete `nothing_special_delete_me.txt` ", - "HELP_12_DESC_22": "The next time you open the Ethereum Wallet application, your account will be listed under \"Accounts\". ", - "HELP_12_DESC_23": "Open Notepad & paste the private key ", - "HELP_12_DESC_24": "Save the file as `nothing_special_delete_me.txt` at `C:` ", - "HELP_12_DESC_25": "Run the command, `geth account import C:\\nothing_special_delete_me.txt` ", - "HELP_12_DESC_26": "This will prompt you to make a new password. This is the password you will use in geth / Ethereum Wallet / Mist whenever you send a transaction, so don't forget it. ", - "HELP_12_DESC_27": "After successful import, delete `nothing_special_delete_me.txt` ", - "HELP_12_DESC_28": "The next time you open the Ethereum Wallet application, your account will be listed under \"Accounts\". ", - "HELP_13_TITLE": "13) What does \"Insufficient funds. Account you try to send transaction from does not have enough funds. Required XXXXXXXXXXXXXXXXXXX and got: XXXXXXXXXXXXXXXX.\" Mean? ", - "HELP_13_DESC_1": "This means you do not have enough Ether in your account to cover the cost of gas. Each transaction (including token and contract transactions) require gas and that gas is paid in Ether. The number displayed is the amount required to cover the cost of the transaction in Wei. Take that number, divide by `1000000000000000000`, and subtract the amount of Ether you were trying to send (if you were attempting to send Ether). This will give you the amount of Ether you need to send to that account to make the transaction. ", - "HELP_14_TITLE": "14) Some sites randomize (seed) the private key generation via mouse movements. MyCrypto.com doesn't do this. Is the random number generation for MyCrypto safe? ", - "HELP_14_DESC_1": "While the mouse moving thing is clever and we understand why people like it, the reality is window.crypto ensures more entropy than your mouse movements. The mouse movements aren't unsafe, it's just that we (and tons of other crypto experiments) believe in window.crypto. In addition, MyCrypto.com can be used on touch devices. Here's a [conversation between an angry redditor and Vitalik Buterin regarding mouse movements v. window.crypto](https://www.reddit.com/r/ethereum/comments/2bilqg/note_there_is_a_paranoid_highsecurity_way_to/cj5sgrm) and here is the [the window.crypto w3 spec](https://dvcs.w3.org/hg/webcrypto-api/raw-file/tip/spec/Overview.html#dfn-GlobalCrypto). ", - "HELP_15_TITLE": "15) Why hasn't the account I just created show up in the blockchain explorer? (ie: etherchain, etherscan) ", - "HELP_15_DESC_1": "Accounts will only show up in a blockchain explorer once the account has activity on it—for example, once you have transferred some Ether to it. ", - "HELP_16_TITLE": "16) How do I check the balance of my account? ", - "HELP_16_DESC_1": "You can use a blockchain explorer like [etherscan.io](https://etherscan.io/). Paste your address into the search bar and it will pull up your address and transaction history. For example, here's what our [donation account](https://etherscan.io/address/0x7cb57b5a97eabe94205c07890be4c1ad31e486a8) looks like on etherscan.io ", - "HELP_17_TITLE": "17) Why isn't my balance showing up when I unlock my wallet? ", - "HELP_17_DESC_1": "This is most likely due to the fact that you are behind a firewall. The API that we use to get the balance and convert said balance is often blocked by firewalls for whatever reason. You will still be able to send transactions, you just need to use a different method to see said balance, like etherscan.io ", - "HELP_18_TITLE": "18) Where is my geth wallet file? ", - "HELP_19_TITLE": "19) Where is my Mist wallet file? ", - "HELP_19_DESC_1": "Mist files are typically found in the file locations above, but it's much easier to open Mist, select \"Accounts\" in the top bar, select \"Backup\", and select \"Accounts\". This will open the folder where your files are stored. ", - "HELP_20_TITLE": "20) Where is my pre-sale wallet file? ", - "HELP_20_DESC_1": "Wherever you saved it. ;) It also was emailed to you, so check there. Look for the file called `\"ethereum_wallet_backup.json\"` and select that file. This wallet file will be encrypted with a password that you created during the purchase of the pre-sale. ", - "HELP_21_TITLE": "21) Couldn't everybody put in random private keys, look for a balance, and send to their own address? ", - "HELP_21_DESC_1": "Short version: yes, but finding an account with a balance would take longer than the universe...so...no. ", - "HELP_21_DESC_2": "Long ELI5 Version: So Ethereum is based on [Public Key Cryptography](https://en.wikipedia.org/wiki/Public-key_cryptography), specifically [Elliptic curve cryptography](https://eprint.iacr.org/2013/734.pdf) which is very widely used, not just in Ethereum. Most servers are protected via ECC. Bitcoin uses the same, as well as SSH and TLS and a lot of other stuff. The Ethereum keys specifically are 256-bit keys, which are stronger than 128-bit and 192-bit, which are also widely used and still considered secure by experts. ", - "HELP_21_DESC_3": "In this you have a private key and a public key. The private key can derive the public key, but the public key cannot be turned back into the private key. The fact that the internet and the world’s secrets are using this cryptography means that if there is a way to go from public key to private key, your lost ether is the least of everyone’s problems. ", - "HELP_21_DESC_4": "Now, that said, YES if someone else has your private key then they can indeed send ether from your account. Just like if someone has your password to your email, they can read and send your email, or the password to your bank account, they could make transfers. You could download the Keystore version of your private key which is the private key that is encrypted with a password. This is like having a password that is also protected by another password. ", - "HELP_21_DESC_5": "And YES, in theory you could just type in a string of 64 hexadecimal characters until you got one that matched. In fact, smart people could write a program to very quickly check random private keys. This is known as \"brute-forcing\" or \"mining\" private keys. People have thought about this long and hard. With a few very high end servers, they may be able to check 1M+ keys / second. However, even checking that many per second would not yield access to make the cost of running those servers even close to worthwhile - it is more likely you, and your great-grandchildren, will die before getting a match. ", - "HELP_21_DESC_6": "If you know anything about Bitcoin, [this will put it in perspective:](https://bitcoin.stackexchange.com/questions/32331/two-people-with-same-public-address-how-will-people-network-know-how-to-deliver) *To illustrate how unlikely this is: suppose every satoshi of every bitcoin ever to be generated was sent to its own unique private keys. The probability that among those keys there could be two that would correspond to the same address is roughly one in 100 quintillion. ", - "HELP_21_DESC_7": "[If you want something a bit more technical:](https://security.stackexchange.com/questions/25375/why-not-use-larger-cipher-keys/25392#25392) *These numbers have nothing to do with the technology of the devices; they are the maximums that thermodynamics will allow. And they strongly imply that brute-force attacks against 256-bit keys will be infeasible until computers are built from something other than matter and occupy something other than space. ", - "HELP_21_DESC_8": "Of course, this all assumes that keys are generated in a truly random way & with sufficient entropy. The keys generated here meet that criteria, as do Jaxx and Mist/geth. The Ethereum wallets are all pretty good. Keys generated by brainwallets do not, as a person's brain is not capable of creating a truly random seed. There have been a number of other issues regarding lack of entropy or seeds not being generated in a truly random way in Bitcoin-land, but that's a separate issue that can wait for another day. ", - "HELP_SECCX_TITLE": "Security - MyCrypto CX ", - "HELP_SECCX_DESC_1": "Where is this extension saving my information? ", - "HELP_SECCX_DESC_2": "The information you store in this Chrome Extension is saved via [chrome.storage](https://chrome.storage/). - this is the same place your passwords are saved when you save your password in Chrome. ", - "HELP_SECCX_DESC_3": "What information is saved? ", - "HELP_SECCX_DESC_4": "The address, nickname, private key is stored in chrome.storage. The private key is encrypted using the password you set when you added the wallet. The nickname and wallet address is not encrypted. ", - "HELP_SECCX_DESC_5": "Why aren't the nickname and wallet address encrypted? ", - "HELP_SECCX_DESC_6": "If we were to encrypt these items, you would need to enter a password each time you wanted to view your account balance or view the nicknames. If this concerns you, we recommend you use MyCrypto.com instead of this Chrome Extension. ", - "HELP_SEC_TITLE": "Security ", - "HELP_SEC_DESC_1": "If one of your first questions is \"Why should I trust these people?\", that is a good thing. Hopefully the following will help ease your fears. ", - "HELP_SEC_DESC_2": "We've been up and running since August 2015. If you search for [\"myetherwallet\" on reddit](https://www.reddit.com/search?q=myetherwallet), you can see numerous people who use us with great success. ", - "HELP_SEC_DESC_3": "We aren't going to take your money or steal your private key(s). There is no malicious code on this site. In fact the \"GENERATE WALLET\" pages are completely client-side. That means that all the code is executed on ** your computer** and it is never saved and transmitted anywhere. ", - "HELP_SEC_DESC_4": "Check the URL -- This site is being served through GitHub and you can see the source code here: [https://github.com/MyCryptoHQ/MyCrypto/tree/gh-pages](https://github.com/MyCryptoHQ/MyCrypto/tree/gh-pages) to [https://www.mycrypto.com](https://www.mycrypto.com). ", - "HELP_SEC_DESC_5": "For generating wallets, you can download the [source code and run it locally](https://download.mycrypto.com/). See #5 above. ", - "HELP_SEC_DESC_6": "Generate a test wallet and check and see what network activity is happening. The easiest way for you to do this is to right click on the page and click \"inspect element\". Go to the \"Network\" tab. Generate a test wallet. You will see there is no network activity. You may see something happening that looks like data:image/gif and data:image/png. Those are the QR codes being generated...on your computer...by your computer. No bytes were transferred. ", - "HELP_SEC_DESC_8": "If you do not feel comfortable using this tool, then by all means, do not use it. We created this tool as a helpful way for people to generate wallets and make transactions without needing to dive into command line or run a full node. Again, feel free to reach out if you have concerns and we will respond as quickly as possible. Thanks! ", - "HELP_FAQ_TITLE": "More Helpful Answers to Frequent Questions ", - "HELP_CONTACT_TITLE": "Ways to Get in Touch", - "ONBOARD_WELCOME_TITLE": "Welcome to MyCrypto.com", - "ONBOARD_WELCOME_CONTENT__1": "Please take some time to understand this for your own safety. 🙏", - "ONBOARD_WELCOME_CONTENT__2": "Your funds will be stolen if you do not heed these warnings.", - "ONBOARD_WELCOME_CONTENT__3": "We know this click-through stuff is annoying. We are sorry.", - "ONBOARD_WELCOME_CONTENT__4": "What is MyCrypto? ", - "ONBOARD_WELCOME_CONTENT__5": "MyCrypto is a free, open-source, client-side interface.", - "ONBOARD_WELCOME_CONTENT__6": "We allow you to interact directly with the blockchain while remaining in full control of your keys & your funds.", - "ONBOARD_WELCOME_CONTENT__7": "**You** and **only you** are responsible for your security.", - "ONBOARD_WELCOME_CONTENT__8": "We cannot recover your funds or freeze your account if you visit a phishing site or lose your private key.", - "ONBOARD_BANK_TITLE": "MyCrypto is not a Bank", - "ONBOARD_BANK_CONTENT__1": "When you open an account with a bank or exchange, they create an account for you in their system.", - "ONBOARD_BANK_CONTENT__2": "The bank keeps track of your personal information, account passwords, balances, transactions and ultimately your money.", - "ONBOARD_BANK_CONTENT__3": "The bank charge fees to manage your account and provide services, like refunding transactions when your card gets stolen.", - "ONBOARD_BANK_CONTENT__4": "The bank allows you to write a check or charge your debit card to send money, go online to check your balance, reset your password, and get a new debit card if you lose it.", - "ONBOARD_BANK_CONTENT__5": "You have an account *with the bank or exchange* and they decide how much money you can send, where you can send it, and how long to hold on a suspicious deposit. All for a fee.", - "ONBOARD_WELCOME_TITLE__ALT": "Introduction", - "ONBOARD_INTERFACE_TITLE": "MyCrypto is an Interface", - "ONBOARD_INTERFACE_CONTENT__1": "When you create an account on MyCrypto you are generating a cryptographic set of numbers: your private key and your public key (address).", - "ONBOARD_INTERFACE_CONTENT__2": "The handling of your keys happens entirely on your computer, inside your browser.", - "ONBOARD_INTERFACE_CONTENT__3": "We never transmit, receive or store your private key, password, or other account information.", - "ONBOARD_INTERFACE_CONTENT__4": "We do not charge a transaction fee.", - "ONBOARD_INTERFACE_CONTENT__5": "You are simply using our **interface** to interact **directly with the blockchain**.", - "ONBOARD_INTERFACE_CONTENT__6": "If you send your *public key (address)* to someone, they can send you ETH or tokens. 👍", - "ONBOARD_INTERFACE_CONTENT__7": "If you send your *private key* to someone, they now have full control of your account. 👎", - "ONBOARD_BANK_TITLE__ALT": "MyCrypto isn't a Bank", - "ONBOARD_BLOCKCHAIN_SKIP": "I already know what a blockchain is...", - "ONBOARD_BLOCKCHAIN_TITLE": "Wait, WTF is a Blockchain?", - "ONBOARD_BLOCKCHAIN_CONTENT__1": "The blockchain is like a huge, global, decentralized spreadsheet.", - "ONBOARD_BLOCKCHAIN_CONTENT__2": "It keeps track of who sent how many coins to whom, and what the balance of every account is.", - "ONBOARD_BLOCKCHAIN_CONTENT__3": "It is stored and maintained by thousands of people (miners) across the globe who have special computers.", - "ONBOARD_BLOCKCHAIN_CONTENT__4": "The blocks in the blockchain are made up of all the individual transactions sent from MyCrypto, MetaMask, Exodus, Mist, Geth, Parity, and everywhere else.", - "ONBOARD_BLOCKCHAIN_CONTENT__5": "When you see your balance on MyCrypto.com or view your transactions on [etherscan.io](https://etherscan.io), you are seeing data on the blockchain, not in our personal systems.", - "ONBOARD_BLOCKCHAIN_CONTENT__6": "Again: **we are not a bank**.", - "ONBOARD_INTERFACE_TITLE__ALT": "MyCrypto is an Interface", - "ONBOARD_WHY_TITLE": "Why are you making me read all this?", - "ONBOARD_WHY_CONTENT__1": "Because we need you to understand that we **cannot**...", - "ONBOARD_WHY_CONTENT__2": "Access your account or send your funds for you X.", - "ONBOARD_WHY_CONTENT__3": "Recover or change your private key.", - "ONBOARD_WHY_CONTENT__4": "Recover or reset your password.", - "ONBOARD_WHY_CONTENT__5": "Reverse, cancel, or refund transactions.", - "ONBOARD_WHY_CONTENT__6": "Freeze accounts.", - "ONBOARD_WHY_CONTENT__7": "**You** and **only you** are responsible for your security.", - "ONBOARD_WHY_CONTENT__8": "Be diligent to keep your private key and password safe. Your private key is sometimes called your mnemonic phrase, keystore file, UTC file, JSON file, wallet file.", - "ONBOARD_WHY_CONTENT__9": "If you lose your private key or password, no one can recover it.", - "ONBOARD_WHY_CONTENT__10": "If you enter your private key on a phishing website, you will have **all your funds taken**.", - "ONBOARD_BLOCKCHAIN_TITLE__ALT": "WTF is a Blockchain?", - "ONBOARD_POINT_TITLE__ALT": "What's the Point of MyCrypto then?", - "ONBOARD_WHYMYC_TITLE": "If MyCrypto can't do those things, what's the point?", - "ONBOARD_WHYMYC_CONTENT__1": "Because that is the point of decentralization and the blockchain.", - "ONBOARD_WHYMYC_CONTENT__2": "You don't have to rely on your bank, government, or anyone else when you want to move your funds.", - "ONBOARD_WHYMYC_CONTENT__3": "You don't have to rely on the security of an exchange or bank to keep your funds safe.", - "ONBOARD_WHYMYC_CONTENT__4": "If you don't find these things valuable, ask yourself why you think the blockchain and cryptocurrencies are valuable. 😉", - "ONBOARD_WHYMYC_CONTENT__5": "If you don't like the sound of this, consider using [Coinbase](https://www.coinbase.com/) or [Blockchain.info](https://blockchain.info/wallet/#/signup). They have more familiar accounts with usernames & passwords.", - "ONBOARD_WHYMYC_CONTENT__6": "If you are scared but want to use MyCrypto, [get a hardware wallet](https://support.mycrypto.com/hardware-wallets/hardware-wallet-recommendations.html)! These keep your keys secure.", - "ONBOARD_WHY_TITLE__ALT": "But...why?", - "ONBOARD_SECURE_TITLE": "How To Protect Yourself & Your Funds", - "ONBOARD_SECURE_1_TITLE": "How To Protect Yourself from Phishers", - "ONBOARD_SECURE_1_CONTENT__1": "Phishers send you a message with a link to a website that looks just like MyCrypto, EtherDelta, Paypal, or your bank, but is not the real website. They steal your information and then steal your money.", - "ONBOARD_SECURE_1_CONTENT__2": "Install [EAL](https://chrome.google.com/webstore/detail/etheraddresslookup/pdknmigbbbhmllnmgdfalmedcmcefdfn) or [MetaMask](https://chrome.google.com/webstore/detail/metamask/nkbihfbeogaeaoehlefnkodbefgpgknn) or [Cryptonite by Metacert](https://chrome.google.com/webstore/detail/cryptonite-by-metacert/keghdcpemohlojlglbiegihkljkgnige) or the [MyCrypto Chrome Extension](https://chrome.google.com/webstore/detail/myetherwallet-cx/nlbmnnijcnlegkjjpcfjclmcfggfefdm) to block malicious websites.", - "ONBOARD_SECURE_1_CONTENT__3": "Always check the URL: `https://www.mycrypto.com`.", - "ONBOARD_SECURE_1_CONTENT__4": "Always make sure the URL bar has `MyCrypto, Inc (US)` in green.", - "ONBOARD_SECURE_1_CONTENT__5": "Do not trust messages or links sent to you randomly via email, Slack, Reddit, Twitter, etc.", - "ONBOARD_SECURE_1_CONTENT__6": "Always navigate directly to a site before you enter information. Do not enter information after clicking a link from a message or email.", - "ONBOARD_SECURE_1_CONTENT__7": "[Install an AdBlocker](https://chrome.google.com/webstore/detail/ublock-origin/cjpalhdlnbpafiamejdnhcphjbkeiagm?hl=en) and do not click ads on your search engine (e.g. Google).", - "ONBOARD_POINT_TITLE__ALT_2": "What's the point?", - "ONBOARD_SECURE_2_TITLE": "How To Protect Yourself from Scams", - "ONBOARD_SECURE_2_CONTENT__1": "People will try to get you to give them money in return for nothing.", - "ONBOARD_SECURE_2_CONTENT__2": "If it is too good to be true, it probably is.", - "ONBOARD_SECURE_2_CONTENT__3": "Research before sending money to someone or some project. Look for information on a variety of websites and forums. Be wary.", - "ONBOARD_SECURE_2_CONTENT__4": "Ask questions when you don't understand something or it doesn't seem right.", - "ONBOARD_SECURE_2_CONTENT__5": "Don't let fear, FUD, or FOMO win over common sense. If something is very urgent, ask yourself \"why?\". It may be to create FOMO or prevent you from doing research.", - "ONBOARD_SECURE_3_TITLE__ALT": "Phuck Phishers", - "ONBOARD_SECURE_3_TITLE": "How To Protect Yourself from Loss", - "ONBOARD_SECURE_3_CONTENT__1": "If you lose your private key or password, it is gone forever. Don't lose it.", - "ONBOARD_SECURE_3_CONTENT__2": "Make a backup of your private key and password. Do NOT just store it on your computer. Print it out on a piece of paper or save it to a USB drive.", - "ONBOARD_SECURE_3_CONTENT__3": "Store this paper or USB drive in a different physical location. A backup is not useful if it is destroyed by a fire or flood along with your laptop.", - "ONBOARD_SECURE_3_CONTENT__4": "Do not store your private key in Dropbox, Google Drive, or other cloud storage. If that account is compromised, your funds will be stolen.", - "ONBOARD_SECURE_3_CONTENT__5": "If you have more than 1-week's worth of pay worth of cryptocurrency, get a hardware wallet. No excuses. It's worth it. I promise.", - "ONBOARD_SECURE_3_CONTENT__6": "[Even more Security Tips!](https://support.mycrypto.com/getting-started/protecting-yourself-and-your-funds.html)", - "ONBOARD_SECURE_2_TITLE__ALT_2": "Screw Scams", - "ONBOARD_FINAL_TITLE__ALT": "One more click & you're done! 🤘", - "ONBOARD_FINAL_TITLE": "Alright, I'm done lecturing you!", - "ONBOARD_FINAL_SUBTITLE": "Sorry for being like this. Onwards!", - "ONBOARD_FINAL_CONTENT__1": "Create a wallet", - "ONBOARD_FINAL_CONTENT__2": "Get a hardware wallet", - "ONBOARD_FINAL_CONTENT__3": "How to Set up MyCrypto + MetaMask", - "ONBOARD_FINAL_CONTENT__4": "How to Run MyCrypto Offline / Locally", - "ONBOARD_FINAL_CONTENT__5": "How to Send via Ledger hardware wallet", - "ONBOARD_FINAL_CONTENT__6": "How to Send via TREZOR hardware wallet", - "ONBOARD_FINAL_CONTENT__7": "How to Send via MetaMask", - "ONBOARD_FINAL_CONTENT__8": "Learn More or Contact Us", - "ONBOARD_FINAL_CONTENT__9": "OMG, please just let me send FFS.", - "ONBOARD_RESUME": "It looks like you didn't finish reading through these slides last time. ProTip: Finish reading through the slides 😉" + "ONBOARD_WELCOME_TITLE": "いらっしゃいませ! MyCrypto.com へようこそ!", + "ONBOARD_WELCOME_CONTENT__1": "安全のためにご理解ください🙏 。", + "ONBOARD_WELCOME_CONTENT__2": "これらの警告をよくご覧いただきませんと、**資産が盗まれる**ことになります。", + "ONBOARD_WELCOME_CONTENT__3": "わずらわしい説明と重々に承知いたしております。申し訳ございません。", + "ONBOARD_WELCOME_CONTENT__4": "MyCryptoって何?", + "ONBOARD_WELCOME_CONTENT__5": "MyCryptoはソースコード公開で、クライアント側にある無料で共用の利用者窓口です。", + "ONBOARD_WELCOME_CONTENT__6": "ブロックチェーン上の暗号鍵と暗号資産の管理を直接行うことができます。", + "ONBOARD_WELCOME_CONTENT__7": "セキュリティの全責任は**自分**に、そして**自分だけ**にあります。", + "ONBOARD_WELCOME_CONTENT__8": "詐欺サイトで騙されたり、暗号鍵や、暗号資産を盗まれてしまうと、誰も取り返したり凍結することはできません。", + "ONBOARD_BANK_TITLE": "MyCryptoは金融機関にあらず ", + "ONBOARD_BANK_CONTENT__1": "取引所などの金融機関に口座を開設すると、その金融機関がシステムの中にあなたの口座を作成します。", + "ONBOARD_BANK_CONTENT__2": "その金融機関は口座のパスワード、残高、取引記録や最終的にあなたの財産を含む個人情報を記録管理します。", + "ONBOARD_BANK_CONTENT__3": "また、カードが盗難に会った際の払い戻しなど、口座の維持管理に必要な手数料を徴収します。", + "ONBOARD_BANK_CONTENT__4": "そのため、送金したり、オンラインで口座の確認をしたり、パスワードや盗難に会ったカードの再発行ができます。", + "ONBOARD_BANK_CONTENT__5": "金融機関や取引所にある口座では、送金額の上限、送金相手の設定、送金前にどれくらいの時間をかけて相手の疑わしさを調べるか、などを独自に決めています。全て手数料が元になっています。", + "ONBOARD_WELCOME_TITLE__ALT": "イントロ", + "ONBOARD_INTERFACE_TITLE": "MyCryptoは、共用の利用者窓口 ", + "ONBOARD_INTERFACE_CONTENT__1": "MyCryptoを使って口座を作る時、一対の暗号のための二つの長い数字、すなわち秘密鍵と公開鍵(アドレス)とが作成されます。", + "ONBOARD_INTERFACE_CONTENT__2": "これらの暗号鍵の操作は全て自分のコンピューターのブラウザの中で行われます。", + "ONBOARD_INTERFACE_CONTENT__3": "私たちは、秘密鍵、パスワードや口座情報の送信や保管を一切いたしません。", + "ONBOARD_INTERFACE_CONTENT__4": "私たちは取引手数料を一切いただきません。", + "ONBOARD_INTERFACE_CONTENT__5": "あなたは、私たちの『利用者窓口』でブロックチェーンを『直接』操作しています。", + "ONBOARD_INTERFACE_CONTENT__6": "公開鍵(アドレス)を相手に教えると、ETHやトークンを受け渡し出来る様になります。👍", + "ONBOARD_INTERFACE_CONTENT__7": "秘密鍵が誰かの手に渡ると、受け取った相手がその口座を自分のものにします。👎", + "ONBOARD_BANK_TITLE__ALT": "MyCryptoは金融機関にあらず ", + "ONBOARD_BLOCKCHAIN_SKIP": "もう知ってるんだけど...", + "ONBOARD_BLOCKCHAIN_TITLE": "ちょっと待って、一体ブロックチェーンって何? ", + "ONBOARD_BLOCKCHAIN_CONTENT__1": "ブロックチェーンは、世界共通の巨大な分散型のスプレッドシートのようなものです。", + "ONBOARD_BLOCKCHAIN_CONTENT__2": "誰が誰にコインをいくら送ったとか、どの口座にいくら、などの履歴を全て持っています。", + "ONBOARD_BLOCKCHAIN_CONTENT__3": "何千もいる特殊なコンピューターを持った人(発掘者)によって記録と管理が行われます。", + "ONBOARD_BLOCKCHAIN_CONTENT__4": "ブロックチェーンのなかのブロックはMyCrypto, MetaMask, Exodus, Mist, Geth, Parityなど至る所で行われている個々の処理や取引で作られています。", + "ONBOARD_BLOCKCHAIN_CONTENT__5": "WMyCryptoや[etherscan.io](https//etherscan.io)で残高や処理を見た時、見えるのは私たちのシステムの中ではなく、ブロックチェーンにあるデータです。", + "ONBOARD_BLOCKCHAIN_CONTENT__6": "もう一度: 『私たちは金融機関ではありません。』", + "ONBOARD_INTERFACE_TITLE__ALT": "MyCryptoは共用の利用者窓口 ", + "ONBOARD_WHY_TITLE": "なんで全部読ませる訳?", + "ONBOARD_WHY_CONTENT__1": "私たちに『出来ない』ものをわかっていただく必要があるためです。", + "ONBOARD_WHY_CONTENT__2": "口座にアクセスしてあなたに代わって何かの送金をすること。", + "ONBOARD_WHY_CONTENT__3": "秘密鍵の回復または変更。", + "ONBOARD_WHY_CONTENT__4": "パスワードのリセットまたは回復。", + "ONBOARD_WHY_CONTENT__5": "処理の巻き戻し、取り消しあるいは金額変更。", + "ONBOARD_WHY_CONTENT__6": "口座の凍結。", + "ONBOARD_WHY_CONTENT__7": "『あなた』が、そして『あなただけ』が自分のセキュリテイの全責任を持ちます。", + "ONBOARD_WHY_CONTENT__8": "秘密鍵とパスワードを大事に保管してください。秘密鍵は、ニーモニックフレーズ、キーストアファイル、UTCファイル、JSONファイル、お財布(ワレット)ファイルとも呼ばれることがあります。", + "ONBOARD_WHY_CONTENT__9": "もし秘密鍵やパスワードを忘れると、誰も回復させることができません。", + "ONBOARD_WHY_CONTENT__10": "もし、あなたが秘密鍵を詐欺サイトで入力すると、**全ての資産**が盗まれてしまいます。", + "ONBOARD_BLOCKCHAIN_TITLE__ALT": "一体ブロックチェーンって何? ", + "ONBOARD_POINT_TITLE__ALT": "それじゃ『MyCrypto』は何ができるの?", + "ONBOARD_WHYMYC_TITLE": "MyCryptoにできないなら、どうすればいいの?", + "ONBOARD_WHYMYC_CONTENT__1": "それがブロックチェーンによる分散化の『本質』だからです。", + "ONBOARD_WHYMYC_CONTENT__2": "自分の資産やお金を動かすときに、金融機関、政府や他の誰かを頼る必要が全くありません。", + "ONBOARD_WHYMYC_CONTENT__3": "自分の資産を守るために、取引所や金融機関のセキュリティを頼る必要もありません。", + "ONBOARD_WHYMYC_CONTENT__4": "もし、こういった事には価値がない、と感じるのでしたら、なぜブロックチェーンや仮想通貨に価値があるのかよーく考えて見てください。", + "ONBOARD_WHYMYC_CONTENT__5": "もし、お気に召さないようでしたら、[Coinbase](https//www.coinbase.com/) や [Blockchain.info](https//blockchain.info/wallet/#/signup)をお使いください。これらは、より身近なIDとパスワードを使用しています。", + "ONBOARD_WHYMYC_CONTENT__6": "もし、少し不安だけれどもMyCryptoを使いたければ、[ハードウェアワレットを入手(英語)](https//support.mycrypto.com/hardware-wallets/hardware-wallet-recommendations.html)!", + "ONBOARD_WHY_TITLE__ALT": "なんで...そんなに?", + "ONBOARD_SECURE_TITLE": "自分自身とその資産の守り方 ", + "ONBOARD_SECURE_1_TITLE": "フィッシング詐欺から自分自身を守る方法 ", + "ONBOARD_SECURE_1_CONTENT__1": "オンライン詐欺師は、MyCrypto、EtherDelta、Paypalやあなたの金融機関のリンクが含まれた個人メッセージを送りつけてきますが、これは偽のサイトです。あなたの情報を盗み、そして資産を盗みます。", + "ONBOARD_SECURE_1_CONTENT__2": "[EAL](https//chrome.google.com/webstore/detail/etheraddresslookup/pdknmigbbbhmllnmgdfalmedcmcefdfn)または[MetaMask](https//support.mycrypto.com/migration/moving-from-private-key-to-metamask.html) or [Cryptonite by Metacert](https//chrome.google.com/webstore/detail/cryptonite-by-metacert/keghdcpemohlojlglbiegihkljkgnige). のクローム拡張を使って、これらの悪意のサイトを遮断してください。", + "ONBOARD_SECURE_1_CONTENT__3": "いつも、必ず、URL:『https//mycrypto.com』を確認してください。", + "ONBOARD_SECURE_1_CONTENT__4": "URLバーに、 `MyCrypto, Inc.`と緑色で表示されていることを必ず確認してください。", + "ONBOARD_SECURE_1_CONTENT__5": "eメール、Slack、Reddit、Twitterやその他から送られてくるメッセージやリンクを信用しないでください。", + "ONBOARD_SECURE_1_CONTENT__6": "情報を入力する前に、常に直接サイトを参照してから閲覧してください。eメールやメッセージのリンクを押してから情報を入力しないでください。", + "ONBOARD_SECURE_1_CONTENT__7": "[AdBlockerを導入](https//chrome.google.com/webstore/detail/ublock-origin/cjpalhdlnbpafiamejdnhcphjbkeiagm?hl=ja)し、サーチエンジンの広告(例:グーグル)をクリックしないでください。", + "ONBOARD_POINT_TITLE__ALT_2": "何ができるの?", + "ONBOARD_SECURE_2_TITLE": "オンライン詐欺師から自分自身を守る方法", + "ONBOARD_SECURE_2_CONTENT__1": "これらの人達は、何の見返りもなしにお金をあげる、と言って近づいてきます。", + "ONBOARD_SECURE_2_CONTENT__2": "もし、ありそうにないほど良い話であれば、きっとその通りです。", + "ONBOARD_SECURE_2_CONTENT__3": "誰か、あるいは何かのプロジェクトに送金する前に、よくお調べください。サイトやフォーラムの情報をよく確かめてください。慎重に。", + "ONBOARD_SECURE_2_CONTENT__4": "A理解できなかったり、正しくないと思われる場合には、直接質問してください。", + "ONBOARD_SECURE_2_CONTENT__5": "不安や懐疑心、流行遅れになる事の恐怖が常識の範囲を越えないように。もし何かとても緊急だと言われたら、なぜだろう、と自分でもう一度考えてみてください。おそらくそれは、時期を逃してしまう恐怖を煽るものか、十分な下調べをさせないための罠です。", + "ONBOARD_SECURE_3_TITLE__ALT": "くたばれ詐欺師", + "ONBOARD_SECURE_3_TITLE": "自分自身を損失から守る方法 ", + "ONBOARD_SECURE_3_CONTENT__1": "もしパスワードや秘密鍵を無くすと、財産を永久に失います。絶対に無くすな。", + "ONBOARD_SECURE_3_CONTENT__2": "秘密鍵やパスワードのバックアップを作成してください。コンピューターに保存したままにしないでください。印刷したり、USBキーに保存してください。", + "ONBOARD_SECURE_3_CONTENT__3": "その紙やUSBキーを別の場所に保管してください。一つだけのバックアップでは、火事や津波などで、ラップトップコンピューターと共に破壊されてしまいます。", + "ONBOARD_SECURE_3_CONTENT__4": "秘密鍵をDropbox、Google Drive、や他のクラウドストレージに保管しないでください。そのアカウント(のセキュリティ)が損なわれた場合に、資産が盗まれてしまいます。", + "ONBOARD_SECURE_3_CONTENT__5": "もし、あなたが自分の1週間以上の給料に値する仮想通貨をお持ちであれば、ハードウェア・ワレットを入手してください。言い訳無用です。それだけの価値があります。私が保証します。", + "ONBOARD_SECURE_3_CONTENT__6": "もし、あなたが自分の1週間以上の給料に値する仮想通貨をお持ちであれば、ハードウェア・ワレットを入手してください。言い訳無用です。それだけの価値があります。私が保証します。", + "ONBOARD_SECURE_2_TITLE__ALT_2": "詐欺をねじ伏せる", + "ONBOARD_FINAL_TITLE__ALT": "もうワンクリックで完了です! 🤘", + "ONBOARD_FINAL_TITLE": "オーライ!講義終了!", + "ONBOARD_FINAL_SUBTITLE": "こんな風で申し訳ありません。次はこれです!", + "ONBOARD_FINAL_CONTENT__1": "お財布を作る", + "ONBOARD_FINAL_CONTENT__2": "ハードウェア・ワレットを手に入れる", + "ONBOARD_FINAL_CONTENT__3": "MyCryptoとMetaMaskの設定方法", + "ONBOARD_FINAL_CONTENT__4": "オフライン/手元でのMyCryptoの使い方", + "ONBOARD_FINAL_CONTENT__5": "Ledgerハードウェア・ワレットでの送出方法", + "ONBOARD_FINAL_CONTENT__6": "TREZORハードウェア・ワレットでの送出方法", + "ONBOARD_FINAL_CONTENT__7": "MetaMaskでの送出方法", + "ONBOARD_FINAL_CONTENT__8": "詳しく調べる、連絡する", + "ONBOARD_FINAL_CONTENT__9": "もう勘弁、早く使わせてください。.", + "ONBOARD_RESUME": "前回スライドを最後までお読みになっていないようです。プロのアドバイス:スライドを全部読みましょう😉", + "DONATE_CURRENCY": "$currencyを寄贈する", + "NOTIFICATION_TYPE_WARNING": "警告", + "GENERATE_WALLET_WARNING": "自分の暗号キーを管理をすることは非常に大きなリスクを伴い、たった一つの誤った操作で修復が不可能な損失を招く恐れがあります。新たに仮想通貨を扱う方々には、 [MetaMask]($metamask_link) の使用、 [Ledger]($ledger_link) あるいは [TREZOR]($trezor_link)といったハードウェアワレットの使用を強くお勧めします。", + "GENERATE_WALLET_HELPLINK_1": "異なるタイプのワレットについて習得し、安全を確保してください。", + "PREFOOTER_WARNING": "MyCrypto.comは暗号キーを保持しません。口座のアクセス、紛失した暗号鍵の保全、パスワードのリセットやトランザクションの巻き戻しは全て不可能です。常に正しいURLである事を確認し、自分暗号鍵を保全してください。", + "PREFOOTER_SECURITY_WARNING": "セキュリティは自分の責任です。", + "FOOTER_ABOUT": "MyCryptoはワレットの生成、ERC-20トークンの操作、そしてブロックチェインをより簡単に操作するための、オープンソース、クライアントサイドのツールです。2015年よりコミュニティによって自らコミュニティために、私たちは人々に力を与える素晴らしい製品の開発に絞って活動しています。", + "FOOTER_SUPPORT": "ヘルプとサポート", + "FOOTER_TEAM": "チーム", + "FOOTER_AFFILIATE_TITLE": "私達とその仲間をご支援ください", + "FOOTER_PRESS": "報道連絡先窓口", + "FOOTER_HELP_AND_DEBUGGING": "ヘルパーとENSデバッグ", + "FOOTER_PARTNERS": "パートナー", + "FOOTER_DONATIONS": "寄付窓口", + "FOOTER_OTHER_APPS": "他のアプリ", + "ETHERSCAMDB": "EtherScamDB", + "ETHER_SECURITY_LOOKUP": "EtherSecurityLookup", + "ETHER_ADDRESS_LOOKUP": "EtherAddressLookup", + "LEDGER_REFERRAL_1": "Ledgerワレットを購入", + "LEDGER_REFERRAL_2": "Ledgerをお持ちでない? 今すぐ一つ発注しましょう!", + "LEDGER_TIP": "**ヒント:** ハードウェアワレットの$networkアプリを使用しており、ブラウザーサポートが設定されていることを事を確認してください。", + "LEDGER_TIMEOUT": "時間切れです", + "LEDGER_WRONG_APP": "デバイス上で誤ったアプリが選択されています。", + "LEDGER_LOCKED": "Ledgerデバイスがロックされています", + "TREZOR_REFERAL": "TREZORを購入", + "KEEPKEY_REFERRAL": "Keepkeyを購入", + "STEELY_REFERRAL": "Steelyを購入", + "ETHERCARD_REFERAL": "ether.cardを購入", + "DISCLAIMER": "免責事項", + "INPUT_CONFIRM_PASSWORD_LABEL": "パスワードを確認", + "INPUT_PASSWORD_LABEL": "パスワー", + "INPUT_USERNAME_LABEL": "ユーザー名", + "INPUT_PASSWORD_PLACEHOLDER": "パスワードは$pass_length 文字以上必要です。", + "GENERATE_KEYSTORE_TITLE": "保存キーファイルを生成", + "GENERATE_KEYSTORE_ACTION": "保存キーファイルの生成", + "GENERATE_KEYSTORE_FAILED": "保存キー生成を失敗、あるいは無効な操作を検出しました。暗号資産のロスを防ぐために、不完全なものはお渡し出ません。ページの再表示や異なるブラウザを用いて再度お試しください。", + "WEAK_PASSWORD": "このパスワードは十分強力ではありません。", + "INVALID_PASSWORD": "パスワードが誤りです", + "GENERATE_MNEMONIC_TITLE": "ニーモニック文節を生成", + "GENERATE_THING": "$thing を生成", + "REGENERATE_MNEMONIC": "フレーズを再生成", + "CONFIRM_MNEMONIC": "単語文節を確認", + "REVEAL_NEXT_MNEMONIC": "次の単語を見せる", + "MNEMONIC_CHOOSE_ADDR": "アドレスを選択", + "MNEMONIC_DESCRIPTION_1": "パスフレーズ中の単語を順にクリックしてください。もし次の単語忘れていたら、下にある'Reveal Next Word'ボタンを、クリックしてください。", + "MNEMONIC_DESCRIPTION_2": "単語を書き留めて下さい。いついかなる時も決してクリップボードにコピーしたり、オンラインサービスに保存しないで下さい。", + "MODAL_BACK": "Back", + "WALLET_UNLOCKING": "アンロック中...", + "HELP_ARTICLE_1": "Nano SでのMycrypto使用方法", + "DECRYPT_PROMPT_SELECT_ADDRESS": "アドレスを選択", + "DECRYPT_DROPDOWN_LABEL": "アドレス群", + "ACTION_1": "確認", + "ACTION_2": "取り消し", + "ACTION_3": "開錠", + "ACTION_4": "戻る", + "ACTION_5": "続き", + "ACTION_6": "次", + "ACTION_7": "ログアウト", + "ACTION_8": "詳細", + "ACTION_9": "現況を照会", + "ACTION_10": "完了", + "ACTION_11": "送出", + "ACTION_12": "ダウンロード", + "ACTION_13": "$thing をダウンロード", + "ACTION_14": "続ける", + "CONFIRM_HARDWARE_WALLET_TRANSACTION": "ハードウェアワレットでトランザクションの状況確認", + "WALLET_LOGOUT_MODAL_TITLE": "間もなくログアウトします。", + "WALLET_LOGOUT_MODAL_DESC": "このページから抜けるとログアウトします。続けてもよろしいですか?", + "CONFIRM_TX_MODAL_TITLE": "トランザクションの確認", + "CONFIRM_TX_SENDING": "これから送出", + "CONFIRM_TX_FEE": "処理料金", + "CONFIRM_TX_TOTAL": "合計", + "CONFIRM_TX_FROM": "送出元", + "CONFIRM_TX_TO": "宛先", + "CONFIRM_TX_VIA_CONTRACT": " $unit のスマート契約経由", + "MNEMONIC_FINAL_DESCRIPTION": "全て完了。ワレットを操作できます。次の4ステップに従い、このワレットを操作して下さい。", + "MNEMONIC_FINAL_STEP_1": "MyCryptoを開く", + "MNEMONIC_FINAL_STEP_2": "アカウントタブに移動する", + "MNEMONIC_FINAL_STEP_3": "ワレットタイプを選択", + "MNEMONIC_FINAL_STEP_4": "パスフレーズを入力", + "MNEMONIC_FINAL_STEP_5": "ファイルとパスワードを入れる", + "VIEW_ONLY_RECENT": "直近のアドレスを選択", + "VIEW_ONLY_ENTER": "アフォレスを入力 (例: 0x4bbeEB066eD09...)", + "GO_TO_ACCOUNT": "アカウントページへ移動", + "INSECURE_WALLET_TYPE_TITLE": "このワレットタイプにはお勧めできません。", + "INSECURE_WALLET_TYPE_DESC": "ウェブサイトから$wallet_typeを操作するのは危険です。このサイト自身が被害に会った場合、あるいは何らかの理由でフィッシングサイトへ誘導された場合には資金を失うことになります。 検討:", + "WALLET_WARNING_CHECK": "どうしても$wallet_typeをオンラインで操作しなければならない時には、URLとSSL証明をしっかり確認して下さい。`ブラウザのURLバーに、https//www.mycrypto.com` と `MyCrypto, Inc (US)` となっているはずです。", + "INSECURE_WALLET_WARNING_1": "私はMetaMaskあるいはハードウェアワレットの使用が可能であり、使用すべきものである事を認めます", + "INSECURE_WALLET_WARNING_2": "私はMyCryptoをダウンロードして手元で使うべきである事を認めます。", + "INSECURE_WALLET_WARNING_3": "私はURLとSSL認証で、これが真性のMyCryptoである事を確認しました。", + "INSECURE_WALLET_RECOMMEND_1": "[MetaMask]($metamask_article) あるいは [Hardware Wallet]($hardware_wallet_article) をお使いください。", + "INSECURE_WALLET_RECOMMEND_2": "[MyCryptoをダウンロードして、手元でオフラインにしてから作動させる]($run_local_article)", + "INSECURE_WALLET_RECOMMEND_3": "こちらをお読みください。 [How to Protect Yourself and Your Funds]($secure_your_eth_article)", + "INSECURE_WALLET_DEPRECATION": "**警告** 操作は近いうちに使用できなくなります。 その場合には [MyCryptoを手元で作動させる](https//download.mycrypto.com) 必要があります。", + "DONT_HAVE_WALLET_PROMPT": "ワレットをお持ちではありませんか?", + "DL_WALLET_WARNING_1": "**失くさないでください!** 一度失くしてしまう元に戻せません。", + "DL_WALLET_WARNING_2": "**隠してください!** 悪質なサイトでこのファイルを使うと、あなたの財産が盗まれます。", + "DL_WALLET_WARNING_3": "**バックアップしてください!** いつかは何億円ものの価値になるとして厳重に管理してください。", + "SWAP_SUPPORT": "何か交換に問題がありましたか。サポートに連絡してください。", + "SWAP_SUPPORT_LINK_BROKEN": "リンクが無効の時はこちらをクリック ERROR", + "SWAP_SEND_TO": "**$origin_amount** を **$origin_label** へ送信", + "SWAP_TIME_LIMIT_WARNING": "時間がかかり過ぎる注文は人手で処理する必要がありますので、そのために必要な時間の分だけコインの受取が遅くなります。", + "SWAP_RECOMMENDED_TX_FEES": "[推奨TX料金を使ってください]($link)", + "ENS_DESCRIPTION": "この [Ethereum Name Service]($ens_docs) (ENS) はイサリアブロックチェインに存在する公開、分散で拡張性に富んだ命名システムです。作成された名前を持つと、イーサの送り先として `ensdomain.eth` を `$example_donation_addr...` のような覚えられないもののの代わりとして使うことができます。", + "ENS_INVALID_INPUT": "7文字以上必要で、特殊記号文字は使用できません", + "ENS_DOMAIN_OWNED": "**$name** はすでに落札、所有されています", + "ENS_DOMAIN_AUCTION": " **$name** のオークションが始りました", + "ENS_DOMAIN_FORBIDDEN": "**$name** は秘匿されています", + "ENS_DOMAIN_OPEN": "**$name** は入札可能です", + "ENS_DOMAIN_REVEAL": " **$name** を開札する時期になりました。", + "ENS_DOMAIN_HIGHEST_BID": "現在の最高入札は ", + "ENS_DOMAIN_PROMPT_REVEAL": "**[開札してください。]($link)**", + "SIGN_MSG_PLACEHOLDER": "この署名したメッセージはこのアドレスの所有者があなたであることを証明することになります。", + "CHECK_TX_STATUS_TITLE": "トランザクションの状況確認", + "CHECK_TX_STATUS_DESCRIPTION_1": "状況確認のため対象のトランザクションハッシュを入力。", + "CHECK_TX_STATUS_DESCRIPTION_2": "トランザクションハッシュは [$block_explorer]($block_explorer_link) でアドレス検索します。", + "CHECK_TX_STATUS_DESCRIPTION_MULTIPLE": "トランザクションハッシュは [$block_explorer]($block_explorer_link) もしくは [$block_explorer_2]($block_explorer_link_2) でアドレス検索します。", + "CHECK_TX_TITLE": "トランザクションが見つかりました", + "TX_STATUS": "ステータス", + "TX_BLOCK_NUMB": "ブロック番号", + "TX_GAS_USED": "ガス消費量", + "VERIFY_TX": "確認 ($block_explorer)", + "SWAP_DEPOSIT_INPUT_LABEL": "デポジット", + "SWAP_RECEIVE_INPUT_LABEL": "受取", + "SWAP_MAX_ERROR": "最大 $rate_max $origin_id", + "SWAP_MIN_ERROR": "最小 $rate_max $origin_id", + "ADD_CUSTOM_TKN_HELP": "難しい?カスタムトークンの追加方法を覚えて下さい。", + "TX_FEE_SCALE_LEFT": "Cheap", + "TX_FEE_SCALE_RIGHT": "Fast", + "CHANGE_WALLET": "違うお財布を選ぶ", + "REQUEST_PAYMENT_QR_TITLE": "支払いQRとコード", + "ACCOUNT": "口座", + "OR": "or", + "SIGNED": "署名済み", + "SELECT_RECENT_TX": "直近のトランザクションから選択", + "SELECT_RECENT_TX_BY_TXHASH": "アドレスでは指定できません。トランザクションハッシュを指定してください。", + "EQUIV_VALS_OFFLINE": "オフラインの間に、使用できなくなりました", + "EQUIV_VALS_TESTNET": "テストネットでこの値は表示されません。", + "EQUIV_VALS_UNSUPPORTED_UNIT": "この単位はサポートされていません。", + "PROMPT_ADD_CUSTOM_TKN": "トークンが見つかりませんか?次でカスタムトークンを追加できます。", + "NEW_CONTRACT_ADDR": "新規契約アドレス", + "TOOLTIP_READ_ONLY_WALLET": "リードオンリーでアドレスを見る。", + "TOOLTIP_MORE_INFO": "追加情報", + "TOOLTIP_INSECURE_WALLET_TYPE": "このワレットタイプは安全ではありません。", + "TOOLTIP_SECURE_WALLET_TYPE": "このワレットタイプは安全です。", + "CUSTOM_NODE_CONFLICT": "既に該当する$conflictedNodeが設定されており、保存すると上書きしてしまいます。", + "CUSTOM_NETWORK": "ネットワーク", + "CUSTOM_NODE_NAME": "Node Name", + "CUSTOM_NETWORK_NAME": "ネットワーク名", + "CUSTOM_NETWORK_CURRENCY": "通貨", + "CUSTOM_NETWORK_CHAIN_ID": "チェインID", + "CUSTOM_NETWORK_URL": "URL", + "CUSTOM_NETWORK_HTTP_AUTH": "HTTP基本認証", + "BROADCAST_TX_TITLE": "署名済みのトランザクションを一斉同報", + "BROADCAST_TX_DESCRIPTION": "署名したトランザクションを上書きして、『トランザクションの送出』をクリック", + "NAME_AUCTION_PROMPT_BID_1": "{name}.ethを入札しますか? ", + "NAME_AUCTION_PROMPT_BID_2": " MyCrypto V3 でここをクリックしてください!", + "OPEN_AUCTION_PROMPT_1": "$name.ethが欲しいですか? ", + "OPEN_AUCTION_PROMPT_2": "MyCrypto v3でオークションに参加して下さい!", + "NAME_OWNED_NAME": "名前", + "NAME_OWNED_LABELHASH": "ラベルハッシュ $name", + "NAME_OWNED_NAMEHASH": "名前ハッシュ $name.eth", + "NAME_OWNED_OWNER": "所有者", + "NAME_OWNED_HIGHEST_BIDDER": "最高額入札者", + "NAME_OWNED_RESOLVED_ADDR": "解決されたアドレス", + "LINK": "[$name]($link)", + "RECENT_TX_NETWORK_EXPLORER": "$network_name ネットワークのエクスプローラー", + "RECENT_TX_HELP": " $network のネットネットワークにあり、のみが表示されます。もし見つからなければ$explorerに全て表示されす。", + "NO_RECENT_TX_FOUND": "直近のMyCryptのトランザクションがありません。$explorerで調べてください。", + "BACK_TO_RECENT_TXS": "直近のトランザクションに戻る", + "WALLET_INFO_UTILITIES": "ユーティリティ", + "SIGN_MSG_SUCCESS": " $addressに対してメッセージで署名を完了しました", + "SIGN_MSG_FAIL": "メッセージ署名エラー $err", + "INSUFFICIENT_FUNDS": "警告:Etherあるいはトークンの残高が不足しておりトランザクションを完了できません。残高を増やすか別ワレットを使用してください。", + "NEW_SWAP": "新規両替の開始", + "ERROR_GAS_LIMIT_LOW": "ガスリミットは少なくとも $limit がこのトランザクションには必要です", + "ERROR_GAS_LIMIT_HIGH": "ガス価格が高すぎます。誤りがなければサポートに連絡してください。", + "VIEW_ADDR": "アドレスを表示", + "X_JSONDESC": "これはパスワードが不要な暗号化されていないJSONフォーマットの秘密鍵です。この暗号化されていないJSONフォーマットの秘密鍵を使えば、誰でもパスワードを使わずに自分のお財布とEtherにアクセスできます。 ", + "CONTRACTS_INTERACT": "操作", + "CONTRACTS_DEPLOY": "展開", + "SELECT_A_THING": "$thingを選択する。", + "NO_CONTRACTS_AVAILABLE": "契約がありません", + "NETWORK_STATUS_ONLINE": "$network ネットワークに接続されました", + "NETWORK_STATUS_OFFLINE": "$network ネットワークから切断されました", + "NETWORK_STATUS_CONNECTING": "接続中..." } } From a71c5daec02cc131d9e9d781445b64319f2d57e1 Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Wed, 9 May 2018 20:32:43 -0400 Subject: [PATCH 10/39] Ignore dist for jest tests (#1737) --- jest_config/jest.config.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/jest_config/jest.config.json b/jest_config/jest.config.json index 47f0627b..ff77ed6a 100644 --- a/jest_config/jest.config.json +++ b/jest_config/jest.config.json @@ -12,7 +12,7 @@ "\\.(css|scss)$": "/jest_config/__mocks__/styleMock.ts", "\\.worker.ts": "/jest_config/__mocks__/workerMock.js" }, - "testPathIgnorePatterns": [], + "testPathIgnorePatterns": ["dist"], "setupFiles": [ "/jest_config/setupJest.js", "/jest_config/__mocks__/localStorage.ts" From 8c7d89f22bedb670051b5010e1e559dfa576ecc8 Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Thu, 10 May 2018 22:34:27 -0400 Subject: [PATCH 11/39] Mark depreciated react lifecycles as unsafe (#1733) --- common/components/BalanceSidebar/EquivalentValues.tsx | 2 +- common/components/BalanceSidebar/TokenBalances/Balances.tsx | 2 +- common/components/CurrentCustomMessage.tsx | 2 +- common/components/GenerateKeystoreModal/index.tsx | 2 +- common/components/SendButtonFactory/OnlineSend.tsx | 2 +- common/components/SubTabs/index.tsx | 2 +- common/components/TXMetaDataPanel/TXMetaDataPanel.tsx | 2 +- common/components/TXMetaDataPanel/components/SimpleGas.tsx | 2 +- common/components/TogglablePassword.tsx | 2 +- common/components/TransactionStatus/TransactionStatus.tsx | 2 +- common/components/WalletDecrypt/WalletDecrypt.tsx | 2 +- .../WalletDecrypt/components/DeterministicWalletsModal.tsx | 2 +- common/components/WalletDecrypt/components/LedgerNano.tsx | 2 +- common/components/WalletDecrypt/components/Mnemonic.tsx | 2 +- common/components/WalletDecrypt/components/Trezor.tsx | 2 +- common/components/renderCbs/UnitConverter.tsx | 2 +- common/components/ui/QRCode.tsx | 4 ++-- common/components/ui/SwapDropdown.tsx | 4 ++-- .../Tabs/CheckTransaction/components/TxHashInput.tsx | 2 +- common/containers/Tabs/CheckTransaction/index.tsx | 2 +- .../components/Interact/components/InteractForm/index.tsx | 2 +- .../Tabs/SendTransaction/components/RequestPayment.tsx | 2 +- .../containers/Tabs/SendTransaction/components/WalletInfo.tsx | 2 +- common/containers/Tabs/Swap/components/CurrencySwap.tsx | 2 +- common/containers/Tabs/Swap/components/CurrentRates.tsx | 2 +- 25 files changed, 27 insertions(+), 27 deletions(-) diff --git a/common/components/BalanceSidebar/EquivalentValues.tsx b/common/components/BalanceSidebar/EquivalentValues.tsx index 431e0aba..e94ffcbb 100644 --- a/common/components/BalanceSidebar/EquivalentValues.tsx +++ b/common/components/BalanceSidebar/EquivalentValues.tsx @@ -88,7 +88,7 @@ class EquivalentValues extends React.Component { }; } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { const { balance, tokenBalances, isOffline, network } = this.props; if ( nextProps.balance !== balance || diff --git a/common/components/BalanceSidebar/TokenBalances/Balances.tsx b/common/components/BalanceSidebar/TokenBalances/Balances.tsx index d5f448da..c951f2d8 100644 --- a/common/components/BalanceSidebar/TokenBalances/Balances.tsx +++ b/common/components/BalanceSidebar/TokenBalances/Balances.tsx @@ -29,7 +29,7 @@ export default class TokenBalances extends React.PureComponent { showCustomTokenForm: false }; - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (nextProps.tokenBalances !== this.props.tokenBalances) { const trackedTokens = nextProps.tokenBalances.reduce((prev, t) => { prev[t.symbol] = !t.balance.isZero(); diff --git a/common/components/CurrentCustomMessage.tsx b/common/components/CurrentCustomMessage.tsx index 016c90e1..9b7b6499 100644 --- a/common/components/CurrentCustomMessage.tsx +++ b/common/components/CurrentCustomMessage.tsx @@ -29,7 +29,7 @@ class CurrentCustomMessageClass extends PureComponent { this.setAddressState(this.props); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.wallet !== nextProps.wallet) { this.setAddressState(nextProps); } diff --git a/common/components/GenerateKeystoreModal/index.tsx b/common/components/GenerateKeystoreModal/index.tsx index 7decf0ae..57a0cc28 100644 --- a/common/components/GenerateKeystoreModal/index.tsx +++ b/common/components/GenerateKeystoreModal/index.tsx @@ -41,7 +41,7 @@ export default class GenerateKeystoreModal extends React.Component } } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (nextProps.privateKey !== this.props.privateKey) { this.setState({ privateKey: nextProps.privateKey || '' }); } diff --git a/common/components/SendButtonFactory/OnlineSend.tsx b/common/components/SendButtonFactory/OnlineSend.tsx index ac44a86e..124ad123 100644 --- a/common/components/SendButtonFactory/OnlineSend.tsx +++ b/common/components/SendButtonFactory/OnlineSend.tsx @@ -64,7 +64,7 @@ class OnlineSendClass extends Component { ); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (nextProps.transactionBroadcasted && this.state.showModal) { this.closeModal(); } diff --git a/common/components/SubTabs/index.tsx b/common/components/SubTabs/index.tsx index 40ccb23b..c9819a0a 100644 --- a/common/components/SubTabs/index.tsx +++ b/common/components/SubTabs/index.tsx @@ -39,7 +39,7 @@ export default class SubTabs extends React.PureComponent { window.removeEventListener('resize', this.handleResize); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { // When new tabs come in, we'll need to uncollapse so that they can // be measured and collapsed again, if needed. if (this.props.tabs !== nextProps.tabs) { diff --git a/common/components/TXMetaDataPanel/TXMetaDataPanel.tsx b/common/components/TXMetaDataPanel/TXMetaDataPanel.tsx index 993fc0d9..da1bfff3 100644 --- a/common/components/TXMetaDataPanel/TXMetaDataPanel.tsx +++ b/common/components/TXMetaDataPanel/TXMetaDataPanel.tsx @@ -76,7 +76,7 @@ class TXMetaDataPanel extends React.Component { } } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if ( (this.props.offline && !nextProps.offline) || this.props.network.unit !== nextProps.network.unit diff --git a/common/components/TXMetaDataPanel/components/SimpleGas.tsx b/common/components/TXMetaDataPanel/components/SimpleGas.tsx index e386af59..84d065d0 100644 --- a/common/components/TXMetaDataPanel/components/SimpleGas.tsx +++ b/common/components/TXMetaDataPanel/components/SimpleGas.tsx @@ -57,7 +57,7 @@ class SimpleGas extends React.Component { this.props.fetchGasEstimates(); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (!this.state.hasSetRecommendedGasPrice && nextProps.gasEstimates) { this.setState({ hasSetRecommendedGasPrice: true }); this.props.setGasPrice(nextProps.gasEstimates.fast.toString()); diff --git a/common/components/TogglablePassword.tsx b/common/components/TogglablePassword.tsx index d5871577..164d4173 100644 --- a/common/components/TogglablePassword.tsx +++ b/common/components/TogglablePassword.tsx @@ -40,7 +40,7 @@ export default class TogglablePassword extends React.PureComponent isVisible: !!this.props.isVisible }; - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.isVisible !== nextProps.isVisible) { this.setState({ isVisible: !!nextProps.isVisible }); } diff --git a/common/components/TransactionStatus/TransactionStatus.tsx b/common/components/TransactionStatus/TransactionStatus.tsx index 0fada1e3..13f417a5 100644 --- a/common/components/TransactionStatus/TransactionStatus.tsx +++ b/common/components/TransactionStatus/TransactionStatus.tsx @@ -31,7 +31,7 @@ class TransactionStatus extends React.Component { this.props.fetchTransactionData(this.props.txHash); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.txHash !== nextProps.txHash) { this.props.fetchTransactionData(nextProps.txHash); } diff --git a/common/components/WalletDecrypt/WalletDecrypt.tsx b/common/components/WalletDecrypt/WalletDecrypt.tsx index b9515f8c..30bd9338 100644 --- a/common/components/WalletDecrypt/WalletDecrypt.tsx +++ b/common/components/WalletDecrypt/WalletDecrypt.tsx @@ -226,7 +226,7 @@ const WalletDecrypt = withRouter( hasAcknowledgedInsecure: false }; - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { // Reset state when unlock is hidden / revealed if (nextProps.hidden !== this.props.hidden) { this.setState({ diff --git a/common/components/WalletDecrypt/components/DeterministicWalletsModal.tsx b/common/components/WalletDecrypt/components/DeterministicWalletsModal.tsx index 80ba7135..7c810d73 100644 --- a/common/components/WalletDecrypt/components/DeterministicWalletsModal.tsx +++ b/common/components/WalletDecrypt/components/DeterministicWalletsModal.tsx @@ -73,7 +73,7 @@ class DeterministicWalletsModalClass extends React.PureComponent { this.getAddresses(); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { const { publicKey, chainCode, seed, dPath } = this.props; if ( nextProps.publicKey !== publicKey || diff --git a/common/components/WalletDecrypt/components/LedgerNano.tsx b/common/components/WalletDecrypt/components/LedgerNano.tsx index e4b3e222..bc46bb63 100644 --- a/common/components/WalletDecrypt/components/LedgerNano.tsx +++ b/common/components/WalletDecrypt/components/LedgerNano.tsx @@ -50,7 +50,7 @@ class LedgerNanoSDecryptClass extends PureComponent { }); }; - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.dPath !== nextProps.dPath && nextProps.dPath) { this.setState({ dPath: nextProps.dPath }); } diff --git a/common/components/WalletDecrypt/components/Mnemonic.tsx b/common/components/WalletDecrypt/components/Mnemonic.tsx index 0e3d30e1..9aef3aac 100644 --- a/common/components/WalletDecrypt/components/Mnemonic.tsx +++ b/common/components/WalletDecrypt/components/Mnemonic.tsx @@ -39,7 +39,7 @@ class MnemonicDecryptClass extends PureComponent { dPath: this.props.dPath }; - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.dPath !== nextProps.dPath) { this.setState({ dPath: nextProps.dPath }); } diff --git a/common/components/WalletDecrypt/components/Trezor.tsx b/common/components/WalletDecrypt/components/Trezor.tsx index 6bc27a81..c8c4bc27 100644 --- a/common/components/WalletDecrypt/components/Trezor.tsx +++ b/common/components/WalletDecrypt/components/Trezor.tsx @@ -41,7 +41,7 @@ class TrezorDecryptClass extends PureComponent { isLoading: false }; - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.dPath !== nextProps.dPath && nextProps.dPath) { this.setState({ dPath: nextProps.dPath }); } diff --git a/common/components/renderCbs/UnitConverter.tsx b/common/components/renderCbs/UnitConverter.tsx index bd63121e..08275d2e 100644 --- a/common/components/renderCbs/UnitConverter.tsx +++ b/common/components/renderCbs/UnitConverter.tsx @@ -29,7 +29,7 @@ const initialState = { userInput: '' }; class UnitConverterClass extends Component { public state: State = initialState; - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { const { userInput } = this.state; if (this.props.decimal !== nextProps.decimal) { diff --git a/common/components/ui/QRCode.tsx b/common/components/ui/QRCode.tsx index aaf7bea3..7c233be8 100644 --- a/common/components/ui/QRCode.tsx +++ b/common/components/ui/QRCode.tsx @@ -15,12 +15,12 @@ interface State { export default class QRCode extends React.PureComponent { public state: State = {}; - public componentWillMount() { + public UNSAFE_componentWillMount() { // Start generating QR codes immediately this.generateQrCode(this.props.data); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { // Regenerate QR codes if props change if (nextProps.data !== this.props.data) { this.generateQrCode(nextProps.data); diff --git a/common/components/ui/SwapDropdown.tsx b/common/components/ui/SwapDropdown.tsx index d1f42194..2f8541c7 100644 --- a/common/components/ui/SwapDropdown.tsx +++ b/common/components/ui/SwapDropdown.tsx @@ -34,7 +34,7 @@ class SwapDropdown extends PureComponent { public dropdown: HTMLDivElement | null; - public componentWillMount() { + public UNSAFE_componentWillMount() { this.buildOptions(this.props.options); document.addEventListener('click', this.handleBodyClick); } @@ -43,7 +43,7 @@ class SwapDropdown extends PureComponent { document.removeEventListener('click', this.handleBodyClick); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.options !== nextProps.options) { this.buildOptions(nextProps.options); } diff --git a/common/containers/Tabs/CheckTransaction/components/TxHashInput.tsx b/common/containers/Tabs/CheckTransaction/components/TxHashInput.tsx index d3c401c4..04cfde6c 100644 --- a/common/containers/Tabs/CheckTransaction/components/TxHashInput.tsx +++ b/common/containers/Tabs/CheckTransaction/components/TxHashInput.tsx @@ -33,7 +33,7 @@ class TxHashInput extends React.Component { this.state = { hash: props.hash || '' }; } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.hash !== nextProps.hash && nextProps.hash) { this.setState({ hash: nextProps.hash }); } diff --git a/common/containers/Tabs/CheckTransaction/index.tsx b/common/containers/Tabs/CheckTransaction/index.tsx index 9001a5f9..8210a7f9 100644 --- a/common/containers/Tabs/CheckTransaction/index.tsx +++ b/common/containers/Tabs/CheckTransaction/index.tsx @@ -34,7 +34,7 @@ class CheckTransaction extends React.Component { } } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { const { network } = this.props; if (network.chainId !== nextProps.network.chainId) { this.setState({ hash: '' }); diff --git a/common/containers/Tabs/Contracts/components/Interact/components/InteractForm/index.tsx b/common/containers/Tabs/Contracts/components/Interact/components/InteractForm/index.tsx index 74b513b7..40427590 100644 --- a/common/containers/Tabs/Contracts/components/Interact/components/InteractForm/index.tsx +++ b/common/containers/Tabs/Contracts/components/Interact/components/InteractForm/index.tsx @@ -63,7 +63,7 @@ class InteractForm extends Component { }; } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { const prevProps = this.props; if (nextProps.currentTo.raw !== prevProps.currentTo.raw) { nextProps.resetState(); diff --git a/common/containers/Tabs/SendTransaction/components/RequestPayment.tsx b/common/containers/Tabs/SendTransaction/components/RequestPayment.tsx index fe615cdf..f12700b4 100644 --- a/common/containers/Tabs/SendTransaction/components/RequestPayment.tsx +++ b/common/containers/Tabs/SendTransaction/components/RequestPayment.tsx @@ -68,7 +68,7 @@ class RequestPayment extends React.Component { this.props.resetTransactionRequested(); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (nextProps.wallet && this.props.wallet !== nextProps.wallet) { this.setWalletAsyncState(nextProps.wallet); } diff --git a/common/containers/Tabs/SendTransaction/components/WalletInfo.tsx b/common/containers/Tabs/SendTransaction/components/WalletInfo.tsx index b3c12804..00e79251 100644 --- a/common/containers/Tabs/SendTransaction/components/WalletInfo.tsx +++ b/common/containers/Tabs/SendTransaction/components/WalletInfo.tsx @@ -30,7 +30,7 @@ export default class WalletInfo extends React.PureComponent { this.setStateFromWallet(this.props.wallet); } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.wallet !== nextProps.wallet) { this.setStateFromWallet(nextProps.wallet); } diff --git a/common/containers/Tabs/Swap/components/CurrencySwap.tsx b/common/containers/Tabs/Swap/components/CurrencySwap.tsx index e177fef9..ede70043 100644 --- a/common/containers/Tabs/Swap/components/CurrencySwap.tsx +++ b/common/containers/Tabs/Swap/components/CurrencySwap.tsx @@ -117,7 +117,7 @@ export default class CurrencySwap extends PureComponent { } } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (nextProps.options !== this.props.options) { this.setState({ options: Object.values(nextProps.options.byId) }); } diff --git a/common/containers/Tabs/Swap/components/CurrentRates.tsx b/common/containers/Tabs/Swap/components/CurrentRates.tsx index 2cdf218f..b56e344a 100644 --- a/common/containers/Tabs/Swap/components/CurrentRates.tsx +++ b/common/containers/Tabs/Swap/components/CurrentRates.tsx @@ -49,7 +49,7 @@ class CurrentRates extends PureComponent { } } - public componentWillReceiveProps(nextProps: Props) { + public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (this.props.isOffline && !nextProps.isOffline) { this.loadRates(); } From f06186e8836878ae68a4ce4b1d1ea7de3ad97978 Mon Sep 17 00:00:00 2001 From: Daniel Kmak Date: Fri, 11 May 2018 05:12:49 +0200 Subject: [PATCH 12/39] Upgrade EAC libs to v1.0.0-beta.3. (#1714) --- common/libs/scheduling/contracts/RequestFactory.ts | 7 +------ common/libs/scheduling/index.ts | 8 +++----- common/selectors/schedule/transaction.ts | 2 -- 3 files changed, 4 insertions(+), 13 deletions(-) diff --git a/common/libs/scheduling/contracts/RequestFactory.ts b/common/libs/scheduling/contracts/RequestFactory.ts index 8524afa7..17ef25ec 100644 --- a/common/libs/scheduling/contracts/RequestFactory.ts +++ b/common/libs/scheduling/contracts/RequestFactory.ts @@ -8,11 +8,10 @@ interface ABIFunc { type address = any; type uint256 = any; -type bytes = any; interface IRequestFactory { validateRequestParams: ABIFunc< - { _addressArgs: address[]; _uintArgs: uint256[]; _callData: bytes; _endowment: uint256 }, + { _addressArgs: address[]; _uintArgs: uint256[]; _endowment: uint256 }, { paramsValidity: boolean[] } >; } @@ -29,10 +28,6 @@ const requestFactoryAbi = [ name: '_uintArgs', type: 'uint256[12]' }, - { - name: '_callData', - type: 'bytes' - }, { name: '_endowment', type: 'uint256' diff --git a/common/libs/scheduling/index.ts b/common/libs/scheduling/index.ts index 8ed90123..d2093aad 100644 --- a/common/libs/scheduling/index.ts +++ b/common/libs/scheduling/index.ts @@ -26,9 +26,9 @@ export const EAC_SCHEDULING_CONFIG = { export const EAC_ADDRESSES = { KOVAN: { - blockScheduler: '0x1afc19a7e642761ba2b55d2a45b32c7ef08269d1', - requestFactory: '0x496e2b6089bde77293a994469b08e9f266d87adb', - timestampScheduler: '0xc6370807f0164bdf10a66c08d0dab1028dbe80a3' + blockScheduler: '0x394ce9fe06c72f18e5a845842974f0c1224b1ff5', + requestFactory: '0x98c128b3d8a0ac240f7b7dd4969ea0ad54f9d330', + timestampScheduler: '0x31bbbf5180f2bd9c213e2e1d91a439677243268a' } }; @@ -163,7 +163,6 @@ export const parseSchedulingParametersValidity = (isValid: boolean[]) => { export const getValidateRequestParamsData = ( toAddress: string, - callData = '', callGas: Wei, callValue: ICurrentValue['value'], windowSize: BN | null, @@ -200,7 +199,6 @@ export const getValidateRequestParamsData = ( gasPrice, requiredDeposit ], - _callData: callData, _endowment: endowment }); }; diff --git a/common/selectors/schedule/transaction.ts b/common/selectors/schedule/transaction.ts index e797d973..902778e9 100644 --- a/common/selectors/schedule/transaction.ts +++ b/common/selectors/schedule/transaction.ts @@ -156,7 +156,6 @@ export const getValidateScheduleParamsCallPayload = ( const deposit = getScheduleDeposit(state); const scheduleTimestamp = getScheduleTimestamp(state); const windowSize = getWindowSize(state); - const callData = getData(state); const scheduleTimezone = getScheduleTimezone(state); const windowStart = getWindowStart(state); @@ -189,7 +188,6 @@ export const getValidateScheduleParamsCallPayload = ( const data = getValidateRequestParamsData( bufferToHex(currentTo.value), - callData.value ? bufferToHex(callData.value) : '', callGasLimit, currentValue.value, windowSizeBlockToMin(windowSize.value, scheduleType.value), From 1e9fe1b394356a040b11b927818cbc719ae90bce Mon Sep 17 00:00:00 2001 From: Daniel Ternyak Date: Thu, 10 May 2018 23:35:10 -0500 Subject: [PATCH 13/39] Update Footer (#1746) --- common/components/Footer/index.tsx | 3 +++ common/config/links.ts | 6 +++++- common/translations/lang/en.json | 2 ++ 3 files changed, 10 insertions(+), 1 deletion(-) diff --git a/common/components/Footer/index.tsx b/common/components/Footer/index.tsx index ecbda241..3af4fe66 100644 --- a/common/components/Footer/index.tsx +++ b/common/components/Footer/index.tsx @@ -82,6 +82,9 @@ export default class Footer extends React.PureComponent { {translateRaw('FOOTER_TEAM')} + + {translateRaw('FOOTER_PRIVACY_POLICY')} +

{translateRaw('FOOTER_ABOUT')}

diff --git a/common/config/links.ts b/common/config/links.ts index 07650aa4..ad9841a9 100644 --- a/common/config/links.ts +++ b/common/config/links.ts @@ -62,8 +62,12 @@ export const productLinks: Link[] = [ text: translateRaw('ETHERSCAMDB') }, { - link: 'https://www.mycrypto.com/helpers.html', + link: 'https://legacy.mycrypto.com/helpers.html', text: translateRaw('FOOTER_HELP_AND_DEBUGGING') + }, + { + link: 'https://hackerone.com/mycrypto', + text: translateRaw('FOOTER_HACKERONE') } ]; diff --git a/common/translations/lang/en.json b/common/translations/lang/en.json index d09f6dd9..2f8bda0f 100644 --- a/common/translations/lang/en.json +++ b/common/translations/lang/en.json @@ -382,6 +382,8 @@ "FOOTER_AFFILIATE_TITLE": "Support Us & Our Friends", "FOOTER_PRESS": "Press Inquiries", "FOOTER_HELP_AND_DEBUGGING": "Helpers & ENS Debugging", + "FOOTER_PRIVACY_POLICY": "Privacy Policy", + "FOOTER_HACKERONE": "HackerOne", "FOOTER_PARTNERS": "Our Partners", "FOOTER_DONATIONS": "Donations", "FOOTER_OTHER_APPS": "Other Apps", From ae2ac4f2c6d7a0e18cad9c12d5768e81ed48ed32 Mon Sep 17 00:00:00 2001 From: William O'Beirne Date: Fri, 11 May 2018 11:15:32 -0400 Subject: [PATCH 14/39] Production Release Changes (#1673) * Remove beta agreement, move modals to Root, and initial work on welcome modal. * Local storage detection for welcome modal * Remove announcement from header. Allow tooltips to point in non-top directions. * Show modal fade at bottom on non-footer modals * Update README * Update all links back to old mycrypto to classic.mycrypto, add footer link too. * Localize welcome modal * Remove release candidate version text, change to legacy.mycrypto instead of classic.mycrypto. * update banner; add hackerone link --- README.md | 6 +- common/Root.tsx | 10 ++- common/components/BetaAgreement/index.scss | 66 ----------------- common/components/BetaAgreement/index.tsx | 73 ------------------- common/components/Footer/index.tsx | 3 - .../Header/components/OnlineStatus.scss | 1 - .../Header/components/OnlineStatus.tsx | 4 +- common/components/WelcomeModal.scss | 26 +++++++ common/components/WelcomeModal.tsx | 62 ++++++++++++++++ common/components/index.ts | 1 - common/components/ui/Modal/ModalBody.tsx | 6 +- common/components/ui/Modal/index.scss | 4 + common/components/ui/Modal/index.tsx | 4 +- common/components/ui/TitleBar.scss | 27 ------- common/components/ui/TitleBar.tsx | 11 --- common/components/ui/Tooltip.scss | 59 +++++++++++++-- common/components/ui/Tooltip.tsx | 6 +- common/components/ui/index.ts | 1 - common/config/data.tsx | 11 +-- common/config/links.ts | 4 + common/containers/OnboardModal/index.tsx | 17 ++--- common/containers/TabSection/WebTemplate.tsx | 3 +- common/sass/mixins.scss | 14 +++- common/sass/variables.scss | 1 + common/sass/variables/tooltips.scss | 4 + common/translations/lang/en.json | 13 +++- common/utils/formatters.ts | 2 +- common/utils/localStorage.ts | 17 +++++ 28 files changed, 233 insertions(+), 223 deletions(-) delete mode 100644 common/components/BetaAgreement/index.scss delete mode 100644 common/components/BetaAgreement/index.tsx create mode 100644 common/components/WelcomeModal.scss create mode 100644 common/components/WelcomeModal.tsx delete mode 100644 common/components/ui/TitleBar.scss delete mode 100644 common/components/ui/TitleBar.tsx create mode 100644 common/sass/variables/tooltips.scss diff --git a/README.md b/README.md index 87d45df8..ae08c10f 100644 --- a/README.md +++ b/README.md @@ -1,8 +1,11 @@ -# MyCrypto Beta RC (VISIT [MyCryptoHQ/mycrypto.com](https://github.com/MyCryptoHQ/mycrypto.com) for the current site)
Just looking to download? Grab our [latest release](https://github.com/MyCryptoHQ/MyCrypto/releases) +# MyCrypto Web & Desktop Apps [![Greenkeeper badge](https://badges.greenkeeper.io/MyCryptoHq/MyCrypto.svg)](https://greenkeeper.io/) [![Coverage Status](https://coveralls.io/repos/github/MyCryptoHQ/MyCrypto/badge.svg?branch=develop)](https://coveralls.io/github/MyCryptoHQ/MyCrypto?branch=develop) +* **Just looking to download?** Grab our [latest release](https://github.com/MyCryptoHQ/MyCrypto/releases). +* **Looking for the old site?** Check out [https://legacy.mycrypto.com](https://legacy.mycrypto.com) or the source at [MyCryptoHQ/mycrypto.com](https://github.com/MyCryptoHQ/mycrypto.com) + ## Running the App This codebase targets Node 8.9.4 (LTS). After `npm install`ing all dependencies (You may be required to install additional system dependencies, due to some node modules relying on them) you can run various commands depending on what you want to do: @@ -132,3 +135,4 @@ npm run test:int Cross browser testing and debugging provided by the very lovely team at BrowserStack. + diff --git a/common/Root.tsx b/common/Root.tsx index 0fe992d7..38d166c6 100644 --- a/common/Root.tsx +++ b/common/Root.tsx @@ -15,8 +15,9 @@ import ErrorScreen from 'components/ErrorScreen'; import PageNotFound from 'components/PageNotFound'; import LogOutPrompt from 'components/LogOutPrompt'; import QrSignerModal from 'containers/QrSignerModal'; +import OnboardModal from 'containers/OnboardModal'; +import WelcomeModal from 'components/WelcomeModal'; import NewAppReleaseModal from 'components/NewAppReleaseModal'; -import { TitleBar } from 'components/ui'; import { Store } from 'redux'; import { pollOfflineStatus, TPollOfflineStatus } from 'actions/config'; import { AppState } from 'reducers'; @@ -104,12 +105,17 @@ class RootClass extends Component { - {process.env.BUILD_ELECTRON && } {routes} {process.env.BUILD_ELECTRON && } + {!process.env.DOWNLOADABLE_BUILD && ( + + + + + )} diff --git a/common/components/BetaAgreement/index.scss b/common/components/BetaAgreement/index.scss deleted file mode 100644 index 6bad52ae..00000000 --- a/common/components/BetaAgreement/index.scss +++ /dev/null @@ -1,66 +0,0 @@ -@import 'common/sass/variables'; -@import 'common/sass/mixins'; - -.BetaAgreement { - @include cover-message; - background: $brand-info; - - &-content { - h2 { - text-align: center; - } - - &-buttons { - padding-top: 20px; - - &-btn { - display: block; - width: 100%; - max-width: 280px; - margin: 0 auto; - border: none; - padding: 0; - transition: $transition; - - &.is-continue { - height: 60px; - line-height: 60px; - font-size: 22px; - background: rgba(#fff, 0.96); - color: $gray-dark; - border-radius: 4px; - margin-bottom: 20px; - - &:hover { - background: #fff; - color: $gray-darker; - } - } - - &.is-reject { - background: none; - color: #fff; - opacity: 0.7; - - &:hover { - opacity: 1; - } - } - } - } - } - - // Fade out - &.is-fading { - pointer-events: none; - opacity: 0; - background: #fff; - transition: all 500ms ease 400ms; - - .BetaAgreement-content { - opacity: 0; - transform: translateY(15px); - transition: all 500ms ease; - } - } -} diff --git a/common/components/BetaAgreement/index.tsx b/common/components/BetaAgreement/index.tsx deleted file mode 100644 index b8472e74..00000000 --- a/common/components/BetaAgreement/index.tsx +++ /dev/null @@ -1,73 +0,0 @@ -import React from 'react'; -import { NewTabLink } from 'components/ui'; -import { discordURL } from 'config'; -import './index.scss'; - -const LS_KEY = 'acknowledged-beta'; - -interface State { - isFading: boolean; - hasAcknowledged: boolean; -} -export default class BetaAgreement extends React.PureComponent<{}, State> { - public state = { - hasAcknowledged: !!localStorage.getItem(LS_KEY), - isFading: false - }; - - public render() { - if (this.state.hasAcknowledged) { - return null; - } - - const isFading = this.state.isFading ? 'is-fading' : ''; - - return ( -
-
-

Welcome to the New MyCrypto Beta Release Candidate!

-

- You are about to use the new MyCrypto Beta Release Candidate. Although this is a release - candidate for production, we encourage caution while using this unreleased version of - MyCrypto. -

-

We hope to move this version of MyCrypto into production in the near future!

-

- Feedback and bug reports are greatly appreciated. You can file issues on our{' '} - - GitHub repository - {' '} - or join our Discord server to discuss the - beta. -

-

Are you sure you would like to continue?

- -
- - -
-
-
- ); - } - - private doContinue = () => { - localStorage.setItem(LS_KEY, 'true'); - this.setState({ isFading: true }); - - setTimeout(() => { - this.setState({ hasAcknowledged: true }); - }, 1000); - }; - - private reject = () => { - window.location.assign('https://mycrypto.com'); - }; -} diff --git a/common/components/Footer/index.tsx b/common/components/Footer/index.tsx index 3af4fe66..027b405e 100644 --- a/common/components/Footer/index.tsx +++ b/common/components/Footer/index.tsx @@ -12,7 +12,6 @@ import React from 'react'; import PreFooter from './PreFooter'; import DisclaimerModal from 'components/DisclaimerModal'; import { NewTabLink } from 'components/ui'; -import OnboardModal from 'containers/OnboardModal'; import './index.scss'; import { translateRaw } from 'translations'; @@ -133,8 +132,6 @@ export default class Footer extends React.PureComponent { - - ); diff --git a/common/components/Header/components/OnlineStatus.scss b/common/components/Header/components/OnlineStatus.scss index 7a3acbc0..c92840f1 100644 --- a/common/components/Header/components/OnlineStatus.scss +++ b/common/components/Header/components/OnlineStatus.scss @@ -12,7 +12,6 @@ .OnlineStatus { position: relative; - top: -2px; width: 12px; height: 12px; text-align: center; diff --git a/common/components/Header/components/OnlineStatus.tsx b/common/components/Header/components/OnlineStatus.tsx index d36e3090..b7dba55d 100644 --- a/common/components/Header/components/OnlineStatus.tsx +++ b/common/components/Header/components/OnlineStatus.tsx @@ -7,8 +7,8 @@ interface Props { } const OnlineStatus: React.SFC = ({ isOffline }) => ( -
- {isOffline ? 'Offline' : 'Online'} +
+ {isOffline ? 'Offline' : 'Online'}
); diff --git a/common/components/WelcomeModal.scss b/common/components/WelcomeModal.scss new file mode 100644 index 00000000..1128dd1c --- /dev/null +++ b/common/components/WelcomeModal.scss @@ -0,0 +1,26 @@ +@import 'common/sass/variables'; + +.WelcomeModal { + font-size: $font-size-bump-more; + + &-logo { + display: block; + max-width: 380px; + margin: $space auto $space * 2; + } + + &-beta { + margin-top: -$space-md; + font-size: $font-size-base; + text-align: center; + } + + &-continue { + display: block; + margin: $space * 2 auto 0; + } + + p, ul { + margin-bottom: $space; + } +} diff --git a/common/components/WelcomeModal.tsx b/common/components/WelcomeModal.tsx new file mode 100644 index 00000000..5d62ba76 --- /dev/null +++ b/common/components/WelcomeModal.tsx @@ -0,0 +1,62 @@ +import React from 'react'; +import translate from 'translations'; +import { Modal, NewTabLink } from 'components/ui'; +import { isLegacyUser, isBetaUser } from 'utils/localStorage'; +import Logo from 'assets/images/logo-mycrypto-transparent.svg'; +import './WelcomeModal.scss'; + +const LS_KEY = 'acknowledged-welcome'; + +interface State { + isOpen: boolean; +} + +export default class WelcomeModal extends React.Component<{}, State> { + public state: State = { + isOpen: false + }; + + public componentDidMount() { + if (isLegacyUser() && !localStorage.getItem(LS_KEY)) { + this.setState({ isOpen: true }); + } + } + + public render() { + return ( + +
+ + {isBetaUser() && ( +

+ 💖 {translate('WELCOME_MODAL_BETA')} 🚀 +

+ )} +

{translate('WELCOME_MODAL_INTRO')}

+
    +
  • {translate('WELCOME_MODAL_FEATURE_1')}
  • +
  • {translate('WELCOME_MODAL_FEATURE_2')}
  • +
  • {translate('WELCOME_MODAL_FEATURE_3')}
  • +
  • {translate('WELCOME_MODAL_FEATURE_4')}
  • +
  • + + {translate('WELCOME_MODAL_FEATURE_5')} + +
  • +
  • {translate('WELCOME_MODAL_FEATURE_MORE')}
  • +
+

{translate('WELCOME_MODAL_LINKS')}

+ + +
+
+ ); + } + + private close = () => { + this.setState({ isOpen: false }); + localStorage.setItem(LS_KEY, 'true'); + }; +} diff --git a/common/components/index.ts b/common/components/index.ts index 4543dfbc..d49f23c4 100644 --- a/common/components/index.ts +++ b/common/components/index.ts @@ -14,7 +14,6 @@ export { default as Header } from './Header'; export { default as Footer } from './Footer'; export { default as BalanceSidebar } from './BalanceSidebar'; export { default as PaperWallet } from './PaperWallet'; -export { default as BetaAgreement } from './BetaAgreement'; export { default as TXMetaDataPanel } from './TXMetaDataPanel'; export { default as WalletDecrypt } from './WalletDecrypt'; export { default as TogglablePassword } from './TogglablePassword'; diff --git a/common/components/ui/Modal/ModalBody.tsx b/common/components/ui/Modal/ModalBody.tsx index 1f4d2fb8..459d39e8 100644 --- a/common/components/ui/Modal/ModalBody.tsx +++ b/common/components/ui/Modal/ModalBody.tsx @@ -3,8 +3,8 @@ import closeIcon from 'assets/images/close.svg'; import { IButton } from 'components/ui/Modal'; interface Props { - title?: string; - children: any; + title?: React.ReactNode; + children: React.ReactNode; modalStyle?: CSSProperties; hasButtons?: number; buttons?: IButton[]; @@ -67,7 +67,7 @@ export default class ModalBody extends React.Component {
(this.modalContent = div as HTMLElement)}> {children} -
+
{hasButtons &&
{this.renderButtons()}
}
diff --git a/common/components/ui/Modal/index.scss b/common/components/ui/Modal/index.scss index 15319619..b7ebbed3 100644 --- a/common/components/ui/Modal/index.scss +++ b/common/components/ui/Modal/index.scss @@ -53,6 +53,10 @@ $m-anim-speed: 400ms; bottom: 4.5rem; left: 50%; transform: translateX(-50%); + + &.has-no-footer { + bottom: 0; + } } &-header { diff --git a/common/components/ui/Modal/index.tsx b/common/components/ui/Modal/index.tsx index 7f2aba92..4ad01c2e 100644 --- a/common/components/ui/Modal/index.tsx +++ b/common/components/ui/Modal/index.tsx @@ -12,9 +12,9 @@ export interface IButton { } interface Props { isOpen?: boolean; - title?: string; + title?: React.ReactNode; disableButtons?: boolean; - children: any; + children: React.ReactNode; buttons?: IButton[]; maxWidth?: number; handleClose(): void; diff --git a/common/components/ui/TitleBar.scss b/common/components/ui/TitleBar.scss deleted file mode 100644 index d6450593..00000000 --- a/common/components/ui/TitleBar.scss +++ /dev/null @@ -1,27 +0,0 @@ -@import 'common/sass/variables'; - -$height: 22px; - -// TODO - Implement styles for custom title bar on all platforms -.TitleBar, -.TitleBarPlaceholder { - display: none; -} - -.TitleBar { - position: fixed; - top: 0; - left: 0; - right: 0; - height: $height; - line-height: $height; - -webkit-user-select: none; - -webkit-app-region: drag; - background: $body-bg; - z-index: $zindex-top; - box-shadow: 0 1px 1px rgba(#000, 0.08); -} - -.TitleBarPlaceholder { - height: $height; -} diff --git a/common/components/ui/TitleBar.tsx b/common/components/ui/TitleBar.tsx deleted file mode 100644 index fe7fdcdd..00000000 --- a/common/components/ui/TitleBar.tsx +++ /dev/null @@ -1,11 +0,0 @@ -import React from 'react'; -import './TitleBar.scss'; - -const TitleBar: React.SFC<{}> = () => ( - -
-
- -); - -export default TitleBar; diff --git a/common/components/ui/Tooltip.scss b/common/components/ui/Tooltip.scss index e6033a23..49cdde1d 100644 --- a/common/components/ui/Tooltip.scss +++ b/common/components/ui/Tooltip.scss @@ -1,9 +1,9 @@ @import 'common/sass/variables'; @import 'common/sass/mixins'; -$tooltip-bg: rgba(#222, 0.95); - .Tooltip { + display: flex; + justify-content: center; position: absolute; top: 0; left: 50%; @@ -14,7 +14,7 @@ $tooltip-bg: rgba(#222, 0.95); pointer-events: none; opacity: 0; visibility: hidden; - transform: translate(-50%, -100%) translateY(-5px); + transform: translate(-50%, -100%) translateY(-$tooltip-start-distance); transition-property: opacity, transform, visibility; transition-duration: 100ms, 100ms, 0ms; transition-delay: 0ms, 0ms, 100ms; @@ -32,7 +32,7 @@ $tooltip-bg: rgba(#222, 0.95); bottom: 0; left: 50%; transform: translate(-50%, 100%); - @include triangle(10px, $tooltip-bg, down); + @include triangle($tooltip-arrow-size * 2, $tooltip-bg, down); } } @@ -46,7 +46,7 @@ $tooltip-bg: rgba(#222, 0.95); border-radius: 2px; &:after { - @include triangle(8px, $tooltip-bg, down); + border-width: $tooltip-arrow-size - 1; } } } @@ -60,8 +60,55 @@ $tooltip-bg: rgba(#222, 0.95); border-radius: 4px; &:after { - @include triangle(12px, $tooltip-bg, down); + border-width: $tooltip-arrow-size + 1; } } } + + // Direction, top is default + &.is-direction-left { + left: 0; + top: 50%; + justify-content: flex-end; + transform: translate(-100%, -50%) translateX(-$tooltip-start-distance); + + > span:after { + bottom: 50%; + right: 0; + left: auto; + transform: translate(100%, 50%); + border-top-color: transparent; + border-left-color: $tooltip-bg; + } + } + + &.is-direction-right { + left: auto; + right: 0; + top: 50%; + justify-content: flex-start; + transform: translate(100%, -50%) translateX($tooltip-start-distance); + + > span:after { + bottom: 50%; + left: 0; + transform: translate(-100%, 50%); + border-top-color: transparent; + border-right-color: $tooltip-bg; + } + } + + &.is-direction-bottom { + top: auto; + bottom: 0; + transform: translate(-50%, 100%) translateY($tooltip-start-distance); + + > span:after { + bottom: auto; + top: 0; + transform: translate(-50%, -100%); + border-top-color: transparent; + border-bottom-color: $tooltip-bg; + } + } } diff --git a/common/components/ui/Tooltip.tsx b/common/components/ui/Tooltip.tsx index 6698f741..00ed8afe 100644 --- a/common/components/ui/Tooltip.tsx +++ b/common/components/ui/Tooltip.tsx @@ -5,13 +5,15 @@ import './Tooltip.scss'; interface Props { children: React.ReactElement | string; size?: 'sm' | 'md' | 'lg'; + direction?: 'top' | 'bottom' | 'left' | 'right'; } -const Tooltip: React.SFC = ({ size, children }) => ( +const Tooltip: React.SFC = ({ size, direction, children }) => (
{children} diff --git a/common/components/ui/index.ts b/common/components/ui/index.ts index 3ab350a2..1f649559 100644 --- a/common/components/ui/index.ts +++ b/common/components/ui/index.ts @@ -11,7 +11,6 @@ export { default as UnitDisplay } from './UnitDisplay'; export { default as Spinner } from './Spinner'; export { default as SwapDropdown } from './SwapDropdown'; export { default as Tooltip } from './Tooltip'; -export { default as TitleBar } from './TitleBar'; export { default as HelpLink } from './HelpLink'; export { default as Input } from './Input'; export { default as TextArea } from './TextArea'; diff --git a/common/config/data.tsx b/common/config/data.tsx index 7637c0d9..90d6266f 100644 --- a/common/config/data.tsx +++ b/common/config/data.tsx @@ -1,16 +1,15 @@ import React from 'react'; // For ANNOUNCEMENT_MESSAGE jsx -import NewTabLink from 'components/ui/NewTabLink'; import { getValues } from '../utils/helpers'; import packageJson from '../../package.json'; import { GasPriceSetting } from 'types/network'; import { makeExplorer } from 'utils/helpers'; +import NewTabLink from 'components/ui/NewTabLink'; export const languages = require('./languages.json'); export const discordURL = 'https://discord.gg/VSaTXEA'; // Displays in the footer -export const VERSION_RAW = packageJson.version; -export const VERSION = `${VERSION_RAW} (Release Candidate 2)`; +export const VERSION = packageJson.version; export const N_FACTOR = 8192; // Bricks the app once this date has been exceeded. Remember to update these 2 @@ -26,10 +25,8 @@ export const VERSION_RC = `${packageJson.version}-RC.0`; export const ANNOUNCEMENT_TYPE = ''; export const ANNOUNCEMENT_MESSAGE = ( - This is a Beta Release Candidate of the new MyCrypto. Please submit any bug reports to our{' '} - GitHub and use{' '} - HackerOne for critical - vulnerabilities. Join the discussion on Discord. + Welcome to the new MyCrypto. We hope you like it! If it's urgent and you need the old site, you + can still use MyCrypto Legacy ); diff --git a/common/config/links.ts b/common/config/links.ts index ad9841a9..3e5c979c 100644 --- a/common/config/links.ts +++ b/common/config/links.ts @@ -47,6 +47,10 @@ export const socialMediaLinks: Link[] = [ ]; export const productLinks: Link[] = [ + { + link: 'https://legacy.mycrypto.com/', + text: translateRaw('OLD_MYCRYPTO') + }, { link: 'https://chrome.google.com/webstore/detail/etheraddresslookup/pdknmigbbbhmllnmgdfalmedcmcefdfn', diff --git a/common/containers/OnboardModal/index.tsx b/common/containers/OnboardModal/index.tsx index 77f4c444..94fcdc50 100644 --- a/common/containers/OnboardModal/index.tsx +++ b/common/containers/OnboardModal/index.tsx @@ -28,9 +28,7 @@ import { SecureSlideThree, FinalSlide } from './components'; - -const ONBOARD_LOCAL_STORAGE_KEY = 'onboardStatus'; -const NUMBER_OF_SLIDES = 10; +import { ONBOARD_LOCAL_STORAGE_KEY, NUMBER_OF_ONBOARD_SLIDES } from 'utils/localStorage'; interface State { isOpen: boolean; @@ -58,7 +56,6 @@ class OnboardModal extends React.Component { public componentDidMount() { const { sessionStarted } = this.props; - const currentSlide = Number(localStorage.getItem(ONBOARD_LOCAL_STORAGE_KEY)) || 0; if (!sessionStarted) { @@ -68,7 +65,7 @@ class OnboardModal extends React.Component { isOpen: true }); } - if (currentSlide > 0 && currentSlide < NUMBER_OF_SLIDES) { + if (currentSlide > 0 && currentSlide < NUMBER_OF_ONBOARD_SLIDES) { this.props.resumeSlide(currentSlide); this.setState({ isOpen: true @@ -90,7 +87,7 @@ class OnboardModal extends React.Component { const firstButtons: IButton[] = [ { - disabled: slideNumber === NUMBER_OF_SLIDES, + disabled: slideNumber === NUMBER_OF_ONBOARD_SLIDES, text: translate('ACTION_6'), type: 'primary', onClick: this.handleNextSlide @@ -115,8 +112,8 @@ class OnboardModal extends React.Component { } ]; - const buttons = slideNumber === NUMBER_OF_SLIDES ? lastButtons : firstButtons; - const steps = new Array(NUMBER_OF_SLIDES).fill({}); + const buttons = slideNumber === NUMBER_OF_ONBOARD_SLIDES ? lastButtons : firstButtons; + const steps = new Array(NUMBER_OF_ONBOARD_SLIDES).fill({}); return (
@@ -158,8 +155,8 @@ class OnboardModal extends React.Component { ]; - if (slides.length !== NUMBER_OF_SLIDES) { - console.log('Slides length do not match const NUMBER_OF_SLIDES'); + if (slides.length !== NUMBER_OF_ONBOARD_SLIDES) { + console.log('Slides length do not match const NUMBER_OF_ONBOARD_SLIDES'); } const currentSlideIndex = this.props.slideNumber - 1; diff --git a/common/containers/TabSection/WebTemplate.tsx b/common/containers/TabSection/WebTemplate.tsx index 7fbb848f..abb6dad3 100644 --- a/common/containers/TabSection/WebTemplate.tsx +++ b/common/containers/TabSection/WebTemplate.tsx @@ -1,6 +1,6 @@ import React, { Component } from 'react'; import { connect } from 'react-redux'; -import { BetaAgreement, Footer, Header } from 'components'; +import { Footer, Header } from 'components'; import { AppState } from 'reducers'; import Notifications from './Notifications'; import OfflineTab from './OfflineTab'; @@ -38,7 +38,6 @@ class WebTemplate extends Component {
-
); } diff --git a/common/sass/mixins.scss b/common/sass/mixins.scss index a995fd16..1e1e770a 100644 --- a/common/sass/mixins.scss +++ b/common/sass/mixins.scss @@ -87,8 +87,20 @@ &:hover .Tooltip { opacity: 1; visibility: visible; - transform: translate(-50%, -100%) translateY(-8px); + transform: translate(-50%, -100%) translateY(-$tooltip-hover-distance); transition-delay: 400ms, 400ms, 300ms; + + &.is-direction-left { + transform: translate(-100%, -50%) translateX(-$tooltip-hover-distance); + } + + &.is-direction-right { + transform: translate(100%, -50%) translateX($tooltip-hover-distance); + } + + &.is-direction-bottom { + transform: translate(-50%, 100%) translateY($tooltip-hover-distance); + } } } diff --git a/common/sass/variables.scss b/common/sass/variables.scss index eb7736a6..c7aa17b7 100644 --- a/common/sass/variables.scss +++ b/common/sass/variables.scss @@ -7,6 +7,7 @@ @import './variables/transitions'; @import './variables/links'; @import './variables/tables'; +@import './variables/tooltips'; @import './variables/buttons'; @import './variables/forms'; @import './variables/zindex'; diff --git a/common/sass/variables/tooltips.scss b/common/sass/variables/tooltips.scss new file mode 100644 index 00000000..21d60481 --- /dev/null +++ b/common/sass/variables/tooltips.scss @@ -0,0 +1,4 @@ +$tooltip-start-distance: 5px; +$tooltip-hover-distance: 9px; +$tooltip-arrow-size: 5px; +$tooltip-bg: rgba(#222, 0.95); diff --git a/common/translations/lang/en.json b/common/translations/lang/en.json index 2f8bda0f..4ba745e7 100644 --- a/common/translations/lang/en.json +++ b/common/translations/lang/en.json @@ -390,6 +390,7 @@ "ETHERSCAMDB": "EtherScamDB", "ETHER_SECURITY_LOOKUP": "EtherSecurityLookup", "ETHER_ADDRESS_LOOKUP": "EtherAddressLookup", + "OLD_MYCRYPTO": "MyCrypto (Legacy Site)", "LEDGER_REFERRAL_1": "Buy a Ledger Wallet", "LEDGER_REFERRAL_2": "Don’t have a Ledger? Order one now!", "LEDGER_TIP": "**Tip:** Make sure you're logged into the $network app on your hardware wallet, and have enabled browser support in the settings", @@ -561,6 +562,16 @@ "APP_UPDATE_TITLE": "New App Version Available", "APP_UPDATE_BODY": "An updated version of the MyCrypto app has just been released! You can upgrade to the new version by downloading it from our GitHub release page.", "APP_UPDATE_CONFIRM": "Get the New Version", - "APP_UPDATE_CANCEL": "Maybe Later" + "APP_UPDATE_CANCEL": "Maybe Later", + "WELCOME_MODAL_BETA": "Thank you for testing the beta! People like you made this launch possible", + "WELCOME_MODAL_INTRO": "Welcome to the all new MyCrypto! We've made some cool new changes to the site that we're excited to show you. Beyond the new and improved look and feel of the site, we've also added a ton of new features:", + "WELCOME_MODAL_FEATURE_1": "Token balance scanner", + "WELCOME_MODAL_FEATURE_2": "Combined Send, Info, and Send Offline tabs", + "WELCOME_MODAL_FEATURE_3": "Parity Signer app integration", + "WELCOME_MODAL_FEATURE_4": "Recent transactions history", + "WELCOME_MODAL_FEATURE_5": "A downloadable desktop app", + "WELCOME_MODAL_FEATURE_MORE": "...and much, much more!", + "WELCOME_MODAL_LINKS": "Help out with any issues you find by [reporting bugs on GitHub](https://github.com/MyCryptoHQ/MyCrypto/issues) or [HackerOne](https://hackerone.com/mycrypto). Need something from the old site, or just miss that clunky feel? We've kept it up as [MyCrypto Legacy](https://legacy.mycrypto.com).", + "WELCOME_MODAL_CONTINUE": "Show me the new site!" } } diff --git a/common/utils/formatters.ts b/common/utils/formatters.ts index f994e948..7df7f3cf 100644 --- a/common/utils/formatters.ts +++ b/common/utils/formatters.ts @@ -113,7 +113,7 @@ export function bytesToHuman(bytes: number) { } export function ensV3Url(name: string) { - return `https://mycrypto.com/?ensname=${name}#ens`; + return `https://legacy.mycrypto.com/?ensname=${name}#ens`; } export function hexToNumber(hex: string) { diff --git a/common/utils/localStorage.ts b/common/utils/localStorage.ts index 5cb8ea24..7b0a0ffb 100644 --- a/common/utils/localStorage.ts +++ b/common/utils/localStorage.ts @@ -4,6 +4,8 @@ import { IWallet, WalletConfig } from 'libs/wallet'; import { AppState } from 'reducers'; export const REDUX_STATE = 'REDUX_STATE'; +export const ONBOARD_LOCAL_STORAGE_KEY = 'onboardStatus'; +export const NUMBER_OF_ONBOARD_SLIDES = 10; export function loadState(): T | undefined { try { @@ -61,3 +63,18 @@ function getWalletConfigKey(wallet: IWallet): string { const address = wallet.getAddressString(); return sha256(`${address}-mycrypto`).toString('hex'); } + +export function isLegacyUser() { + // All devs are legacy users! + const oldLSValue = localStorage.getItem('gasPrice') || process.env.NODE_ENV !== 'production'; + const onboardProgress = localStorage.getItem(ONBOARD_LOCAL_STORAGE_KEY); + + if (oldLSValue && onboardProgress && parseInt(onboardProgress, 10) >= NUMBER_OF_ONBOARD_SLIDES) { + return true; + } + return false; +} + +export function isBetaUser() { + return !!localStorage.getItem('acknowledged-beta'); +} From 6c84654172a72aa99262d424214d4e6b44281a36 Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Fri, 11 May 2018 15:02:50 -0400 Subject: [PATCH 15/39] Increase shepherd thresholds (#1736) * Mark depreciated react lifecycles as unsafe * Rename mycrypto-shepherd -> myc-shepherd, increase thresholds * swap myc-shepherd for mycrypto-shepherd --- common/libs/nodes/index.ts | 6 +++--- package.json | 2 +- yarn.lock | 18 +++++++++++------- 3 files changed, 15 insertions(+), 11 deletions(-) diff --git a/common/libs/nodes/index.ts b/common/libs/nodes/index.ts index e896573a..43fc7c9a 100644 --- a/common/libs/nodes/index.ts +++ b/common/libs/nodes/index.ts @@ -11,7 +11,7 @@ export const makeProviderConfig = (options: DeepPartial = {}): const defaultConfig: IProviderConfig = { concurrency: 2, network: 'ETH', - requestFailureThreshold: 3, + requestFailureThreshold: 10, supportedMethods: { getNetVersion: true, ping: true, @@ -30,7 +30,7 @@ export const makeProviderConfig = (options: DeepPartial = {}): signMessage: true, sendTransaction: true }, - timeoutThresholdMs: 5000 + timeoutThresholdMs: 10000 }; return { @@ -45,7 +45,7 @@ export const makeProviderConfig = (options: DeepPartial = {}): let shepherdProvider: INode; shepherd - .init() + .init({ queueTimeout: 10000 }) .then( provider => (shepherdProvider = (new Proxy(provider, tokenBalanceHandler) as any) as INode) ); diff --git a/package.json b/package.json index 5ab038ad..bd44f6e7 100644 --- a/package.json +++ b/package.json @@ -30,7 +30,7 @@ "lodash": "4.17.5", "moment": "2.22.1", "moment-timezone": "0.5.14", - "mycrypto-shepherd": "1.2.0", + "mycrypto-shepherd": "1.3.1", "normalizr": "3.2.4", "qrcode": "1.2.0", "qrcode.react": "0.8.0", diff --git a/yarn.lock b/yarn.lock index e2292ffc..056c6cf2 100644 --- a/yarn.lock +++ b/yarn.lock @@ -43,9 +43,9 @@ core-js "^2.5.3" regenerator-runtime "^0.11.1" -"@parity/qr-signer@0.1.1": - version "0.1.1" - resolved "https://registry.yarnpkg.com/@parity/qr-signer/-/qr-signer-0.1.1.tgz#3fd268bba845c37bc06d9aa0abe2a1050d8b73e1" +"@parity/qr-signer@0.2.0": + version "0.2.0" + resolved "https://registry.yarnpkg.com/@parity/qr-signer/-/qr-signer-0.2.0.tgz#5c6c41c03265608c117346f0a9d7ab89699352fa" dependencies: qrcode-generator "1.3.1" react-qr-reader "2.0.1" @@ -204,6 +204,10 @@ dependencies: redux "^3.6.0" +"@types/semver@5.5.0": + version "5.5.0" + resolved "https://registry.yarnpkg.com/@types/semver/-/semver-5.5.0.tgz#146c2a29ee7d3bae4bf2fcb274636e264c813c45" + "@types/url-search-params@^0.10.1": version "0.10.1" resolved "https://registry.yarnpkg.com/@types/url-search-params/-/url-search-params-0.10.1.tgz#a5b1f30bf4adad70b103e22c3e1a478869b51be7" @@ -7422,9 +7426,9 @@ mute-stream@0.0.7: version "0.0.7" resolved "https://registry.yarnpkg.com/mute-stream/-/mute-stream-0.0.7.tgz#3075ce93bc21b8fab43e1bc4da7e8115ed1e7bab" -mycrypto-shepherd@1.2.0: - version "1.2.0" - resolved "https://registry.yarnpkg.com/mycrypto-shepherd/-/mycrypto-shepherd-1.2.0.tgz#856a7b250f187dc90508cd7b98ed2dba95c6ea0b" +mycrypto-shepherd@1.3.1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/mycrypto-shepherd/-/mycrypto-shepherd-1.3.1.tgz#901d86e2278c08fd9a3a0aee7a13dea971dbbc38" dependencies: "@types/jest" "^22.2.2" "@types/node" "^9.6.2" @@ -9977,7 +9981,7 @@ semver-truncate@^1.0.0: dependencies: semver "^5.3.0" -"semver@2 || 3 || 4 || 5", semver@^5.0.1, semver@^5.0.3, semver@^5.1.0, semver@^5.3.0, semver@^5.4.1, semver@^5.5.0: +"semver@2 || 3 || 4 || 5", semver@5.5.0, semver@^5.0.1, semver@^5.0.3, semver@^5.1.0, semver@^5.3.0, semver@^5.4.1, semver@^5.5.0: version "5.5.0" resolved "https://registry.yarnpkg.com/semver/-/semver-5.5.0.tgz#dc4bbc7a6ca9d916dee5d43516f0092b58f7b8ab" From fee8d630885e1864a131c46e54fcd221dbf880d5 Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Fri, 11 May 2018 15:07:51 -0400 Subject: [PATCH 16/39] Parse bytes types to a hex buffer (#1731) * Parse bytes types to a hex buffer * Revert yarn lock changes --- common/libs/contracts/ABIFunction.ts | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/common/libs/contracts/ABIFunction.ts b/common/libs/contracts/ABIFunction.ts index 2453368b..54502edd 100644 --- a/common/libs/contracts/ABIFunction.ts +++ b/common/libs/contracts/ABIFunction.ts @@ -95,8 +95,12 @@ export default class AbiFunction { return mapppedType ? mapppedType(value) : BN.isBN(value) ? value.toString() : value; }; - private parsePreEncodedValue = (_: string, value: any) => - BN.isBN(value) ? value.toString() : value; + private parsePreEncodedValue = (type: string, value: any) => { + if (type === 'bytes') { + return Buffer.from(value, 'hex'); + } + return BN.isBN(value) ? value.toString() : value; + }; private makeFuncParams = () => this.inputs.reduce((accumulator, currInput) => { From 40e60932025f924ea61b9e280a577e9cf28fb1ee Mon Sep 17 00:00:00 2001 From: Daniel Ternyak Date: Sat, 12 May 2018 18:26:18 -0500 Subject: [PATCH 17/39] Add private key deprecation information (#1748) * add copy/link to PR for deprecation * Add Parity Signer and MetaMask to pkey warning --- common/translations/lang/en.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/common/translations/lang/en.json b/common/translations/lang/en.json index 4ba745e7..1eb762d3 100644 --- a/common/translations/lang/en.json +++ b/common/translations/lang/en.json @@ -466,7 +466,7 @@ "INSECURE_WALLET_RECOMMEND_1": "Using [MetaMask]($metamask_article) or a [Hardware Wallet]($hardware_wallet_article) to access your wallet", "INSECURE_WALLET_RECOMMEND_2": "[Downloading MyCrypto and running it offline & locally]($download_mycrypto)", "INSECURE_WALLET_RECOMMEND_3": "Reading [How to Protect Yourself and Your Funds]($secure_your_eth_article)", - "INSECURE_WALLET_DEPRECATION": "**Warning:** Accessing this type of wallet via the web interface will be disabled soon. You’ll need to [run MyCrypto locally](https://download.mycrypto.com) going forward.", + "INSECURE_WALLET_DEPRECATION": "**Warning:** Accessing this type of wallet via the web interface will be disabled soon. You’ll need to run [MyCrypto Desktop](https://download.mycrypto.com), [Parity Signer](https://paritytech.io/send-transactions-with-mycrypto-beta-and-parity-signer/), or [MetaMask](https://metamask.io/) going forward. You can learn more [here](https://github.com/MyCryptoHQ/MyCrypto/pull/1466).", "DONT_HAVE_WALLET_PROMPT": "Don’t have a wallet?", "DL_WALLET_WARNING_1": "**Don't lose it!** It can't be recovered if you lose it.", "DL_WALLET_WARNING_2": "**Don't share it!** Your funds will be stolen if you use this file on a malicious site.", From 12864b30d9b654e92b563a3127e5d9ac382300c9 Mon Sep 17 00:00:00 2001 From: James Prado Date: Sat, 12 May 2018 19:27:34 -0400 Subject: [PATCH 18/39] Add custom meta tags for social-media seo (#1755) --- common/assets/images/link-preview.png | Bin 0 -> 61296 bytes common/index.html | 25 ++++++++++++++++-------- webpack_config/config.js | 19 +++++++++++------- webpack_config/makeConfig.js | 27 +++++++++++++++----------- 4 files changed, 45 insertions(+), 26 deletions(-) create mode 100644 common/assets/images/link-preview.png diff --git a/common/assets/images/link-preview.png b/common/assets/images/link-preview.png new file mode 100644 index 0000000000000000000000000000000000000000..8cffb12f08792a9a251f8fe0bb58c15b539dc020 GIT binary patch literal 61296 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 literal 0 HcmV?d00001 diff --git a/common/index.html b/common/index.html index 9abc0108..724de805 100644 --- a/common/index.html +++ b/common/index.html @@ -5,8 +5,16 @@ MyCrypto - + + + + + + + + + @@ -32,16 +40,17 @@

Your Browser is Out of Date

- MyCrypto requires certain features that your browser doesn't offer. Your browser may also be missing security updates - that could open you up to vulnerabilities. Please update your browser, or switch to one of the following browsers - to continue using MyCrypto. + MyCrypto requires certain features that your browser doesn't offer. Your browser may also be missing security updates that + could open you up to vulnerabilities. Please update your browser, or switch to one of the following browsers to continue + using MyCrypto.

- MyCrypto requires certain features that your browser doesn't offer. Please use your device's default browser, or switch - to a laptop or desktop computer to continue using MyCrypto. + MyCrypto requires certain features that your browser doesn't offer. Please use your device's default browser, or switch to + a laptop or desktop computer to continue using MyCrypto.

- + Firefox @@ -97,4 +106,4 @@ - + \ No newline at end of file diff --git a/webpack_config/config.js b/webpack_config/config.js index 8a5a0a13..9ae8701d 100644 --- a/webpack_config/config.js +++ b/webpack_config/config.js @@ -9,13 +9,22 @@ const paths = { static: path.join(__dirname, '../static'), electron: path.join(__dirname, '../electron-app'), shared: path.join(__dirname, '../shared'), - modules: path.join(__dirname, '../node_modules'), -} + modules: path.join(__dirname, '../node_modules') +}; module.exports = { // Configuration port: process.env.HTTPS ? 3443 : 3000, title: 'MyCrypto', + // description < 200 characters + description: 'MyCrypto is a free, open-source interface for interacting with the blockchain.', + url: 'https://mycrypto.com/', + type: 'website', + // img < 5MB + img: path.join(paths.assets, 'images/link-preview.png'), + twitter: { + creator: '@MyCrypto' + }, path: paths, // Typescript rule config @@ -31,11 +40,7 @@ module.exports = { // File resolution resolve: { extensions: ['.ts', '.tsx', '.js', '.css', '.json', '.scss'], - modules: [ - paths.src, - paths.modules, - paths.root, - ] + modules: [paths.src, paths.modules, paths.root] }, // Vendor modules diff --git a/webpack_config/makeConfig.js b/webpack_config/makeConfig.js index 0126e1fa..365c2b09 100644 --- a/webpack_config/makeConfig.js +++ b/webpack_config/makeConfig.js @@ -77,18 +77,11 @@ module.exports = function(opts = {}) { rules.push( { test: /\.css$/, - use: [ - MiniCSSExtractPlugin.loader, - 'css-loader' - ] + use: [MiniCSSExtractPlugin.loader, 'css-loader'] }, { test: /\.scss$/, - use: [ - MiniCSSExtractPlugin.loader, - 'css-loader', - sassLoader - ] + use: [MiniCSSExtractPlugin.loader, 'css-loader', sassLoader] } ); } else { @@ -161,9 +154,17 @@ module.exports = function(opts = {}) { // ==================== const plugins = [ new HtmlWebpackPlugin({ - title: config.title, template: path.resolve(config.path.src, 'index.html'), - inject: true + inject: true, + title: config.title, + appDescription: config.description, + appUrl: config.url, + image: config.img, + type: config.type, + twitter: { + site: config.twitter.creator, + creator: config.twitter.creator + } }), new CopyWebpackPlugin([ @@ -171,6 +172,10 @@ module.exports = function(opts = {}) { from: config.path.static, // to the root of dist path to: './' + }, + { + from: path.resolve(config.path.assets, 'images/link-preview.png'), + to: './common/assets/images' } ]), From 5baa356ee1f4e5f1095bc7735108c56201adb46d Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Sat, 12 May 2018 19:28:03 -0400 Subject: [PATCH 19/39] Fix integer checks for custom token decimals (#1753) --- .../BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/common/components/BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx b/common/components/BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx index c355f25c..f17b9c50 100644 --- a/common/components/BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx +++ b/common/components/BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx @@ -109,7 +109,7 @@ export default class AddCustomTokenForm extends React.PureComponent Date: Sat, 12 May 2018 23:58:58 -0400 Subject: [PATCH 20/39] Automatically track custom tokens (#1769) --- .../BalanceSidebar/TokenBalances/Balances.tsx | 24 ++++++++++++++++--- .../BalanceSidebar/TokenBalances/TokenRow.tsx | 13 ++++++---- 2 files changed, 29 insertions(+), 8 deletions(-) diff --git a/common/components/BalanceSidebar/TokenBalances/Balances.tsx b/common/components/BalanceSidebar/TokenBalances/Balances.tsx index c951f2d8..c21f35d1 100644 --- a/common/components/BalanceSidebar/TokenBalances/Balances.tsx +++ b/common/components/BalanceSidebar/TokenBalances/Balances.tsx @@ -20,7 +20,7 @@ interface TrackedTokens { } interface State { - trackedTokens: { [symbol: string]: boolean }; + trackedTokens: TrackedTokens; showCustomTokenForm: boolean; } export default class TokenBalances extends React.PureComponent { @@ -32,7 +32,7 @@ export default class TokenBalances extends React.PureComponent { public UNSAFE_componentWillReceiveProps(nextProps: Props) { if (nextProps.tokenBalances !== this.props.tokenBalances) { const trackedTokens = nextProps.tokenBalances.reduce((prev, t) => { - prev[t.symbol] = !t.balance.isZero(); + prev[t.symbol] = !t.balance.isZero() || t.custom; return prev; }, {}); this.setState({ trackedTokens }); @@ -45,7 +45,7 @@ export default class TokenBalances extends React.PureComponent { let bottom; let help; - if (tokenBalances.length && !hasSavedWalletTokens) { + if (tokenBalances.length && !hasSavedWalletTokens && !this.onlyCustomTokens()) { help = 'Select which tokens you would like to keep track of'; bottom = (
@@ -134,6 +134,24 @@ export default class TokenBalances extends React.PureComponent { }); }; + /** + * + * @description Checks if all currently tracked tokens are custom + * @private + * @returns + * @memberof TokenBalances + */ + private onlyCustomTokens() { + const tokenMap = this.props.tokenBalances.reduce<{ [key: string]: TokenBalance }>( + (acc, cur) => ({ ...acc, [cur.symbol]: cur }), + {} + ); + + return Object.keys(this.state.trackedTokens).reduce( + (prev, tokenName) => tokenMap[tokenName].custom && prev, + true + ); + } private addCustomToken = (token: Token) => { this.props.onAddCustomToken(token); this.setState({ showCustomTokenForm: false }); diff --git a/common/components/BalanceSidebar/TokenBalances/TokenRow.tsx b/common/components/BalanceSidebar/TokenBalances/TokenRow.tsx index f93cdff0..8fccce3b 100644 --- a/common/components/BalanceSidebar/TokenBalances/TokenRow.tsx +++ b/common/components/BalanceSidebar/TokenBalances/TokenRow.tsx @@ -30,11 +30,14 @@ export default class TokenRow extends React.PureComponent { return ( - {this.props.toggleTracked && ( - - - - )} + {/* Only allow to toggle tracking on non custom tokens + because the user can just remove the custom token instead */} + {!this.props.custom && + this.props.toggleTracked && ( + + + + )} Date: Sun, 13 May 2018 00:20:36 -0400 Subject: [PATCH 21/39] Do not allow custom tokens to be added with the same address (#1770) * Do not allow custom tokens to be added with the same address * prettify --- .../TokenBalances/AddCustomTokenForm.tsx | 53 +++++++++++-------- 1 file changed, 31 insertions(+), 22 deletions(-) diff --git a/common/components/BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx b/common/components/BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx index f17b9c50..bfb20213 100644 --- a/common/components/BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx +++ b/common/components/BalanceSidebar/TokenBalances/AddCustomTokenForm.tsx @@ -16,8 +16,13 @@ interface IGenerateSymbolLookup { [tokenSymbol: string]: boolean; } +interface IGenerateAddressLookup { + [address: string]: boolean; +} + interface State { tokenSymbolLookup: IGenerateSymbolLookup; + tokenAddressLookup: IGenerateAddressLookup; address: string; symbol: string; decimal: string; @@ -25,20 +30,13 @@ interface State { export default class AddCustomTokenForm extends React.PureComponent { public state: State = { - tokenSymbolLookup: {}, + tokenSymbolLookup: this.generateSymbolLookup(), + tokenAddressLookup: this.generateAddressMap(), address: '', symbol: '', decimal: '' }; - constructor(props: Props) { - super(props); - this.state = { - ...this.state, - tokenSymbolLookup: this.generateSymbolLookup(props.allTokens) - }; - } - public render() { const { address, symbol, decimal } = this.state; const errors = this.getErrors(); @@ -76,7 +74,7 @@ export default class AddCustomTokenForm extends React.PureComponent - {typeof errors[field.name] === 'string' && ( + {errors[field.name] && (
{errors[field.name]}
)} @@ -106,14 +104,19 @@ export default class AddCustomTokenForm extends React.PureComponent { - prev[tk.symbol] = true; - return prev; - }, - {} as IGenerateSymbolLookup - ); + private generateSymbolLookup() { + return this.tknArrToMap('symbol'); + } + + private generateAddressMap() { + return this.tknArrToMap('address'); + } + + private tknArrToMap(key: Exclude) { + const tokens = this.props.allTokens; + return tokens.reduce<{ [k: string]: boolean }>((prev, tk) => { + prev[tk[key]] = true; + return prev; + }, {}); } } From b58e278ac5d4e6f97d5b892ab273f826505e3022 Mon Sep 17 00:00:00 2001 From: Daniel Ternyak Date: Sun, 13 May 2018 13:32:45 -0500 Subject: [PATCH 22/39] Exclude components from coverage reports (#1774) * add 'components' and 'containers' dirs to coverage excluded paths * add vendor and spec to excluded coverage paths * trigger mycryptobuild --- jest_config/jest.config.json | 1 + 1 file changed, 1 insertion(+) diff --git a/jest_config/jest.config.json b/jest_config/jest.config.json index ff77ed6a..1cfa2b5f 100644 --- a/jest_config/jest.config.json +++ b/jest_config/jest.config.json @@ -13,6 +13,7 @@ "\\.worker.ts": "/jest_config/__mocks__/workerMock.js" }, "testPathIgnorePatterns": ["dist"], + "coveragePathIgnorePatterns": ["components", "containers", "vendor", "spec"], "setupFiles": [ "/jest_config/setupJest.js", "/jest_config/__mocks__/localStorage.ts" From 32416469e41e43ee869acdc23324839f317d59e5 Mon Sep 17 00:00:00 2001 From: Daniel Ternyak Date: Sun, 13 May 2018 14:09:19 -0500 Subject: [PATCH 23/39] Adjust privacy policy URL (#1775) --- common/components/Footer/index.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/common/components/Footer/index.tsx b/common/components/Footer/index.tsx index 027b405e..fd64d91e 100644 --- a/common/components/Footer/index.tsx +++ b/common/components/Footer/index.tsx @@ -81,7 +81,7 @@ export default class Footer extends React.PureComponent { {translateRaw('FOOTER_TEAM')} - + {translateRaw('FOOTER_PRIVACY_POLICY')}
From 8d27d0ba4df4c580ae58a055cc04be03686226cc Mon Sep 17 00:00:00 2001 From: HenryNguyen5 Date: Sun, 13 May 2018 15:24:50 -0400 Subject: [PATCH 24/39] Improve form validation (#1772) * Change gas price validation to be string input based * Change sanitization to use Nunber * Have validators use Number over parseFloat * Fix css validation class * Add valid css to address field * Add data field validation * Remove unused import * Fix button being hidden on inputs * Dead code removal * Unify textarea and input class validation * Adjust validity styling to only apply after a value has been inputted * Do not pass custom props to DOM --- common/components/AddressField.tsx | 2 +- common/components/AmountField.tsx | 4 +- .../TokenBalances/AddCustomTokenForm.tsx | 5 +-- .../components/Body/components/Details.tsx | 7 ++- .../CustomNodeModal/CustomNodeModal.tsx | 26 +++-------- common/components/DataField.tsx | 4 +- .../DataFieldFactory/DataFieldFactory.tsx | 2 +- .../DataFieldFactory/DataInputFactory.tsx | 11 ++--- common/components/GasLimitField.tsx | 2 +- common/components/NonceField.scss | 2 +- common/components/NonceField.tsx | 3 +- .../components/AdvancedGas.tsx | 9 ++-- .../TXMetaDataPanel/components/FeeSummary.tsx | 3 ++ common/components/TogglablePassword.tsx | 6 ++- .../components/DeterministicWalletsModal.tsx | 2 +- .../WalletDecrypt/components/Keystore.tsx | 5 +-- .../WalletDecrypt/components/Mnemonic.tsx | 2 + .../WalletDecrypt/components/PrivateKey.tsx | 2 +- .../WalletDecrypt/components/ViewOnly.tsx | 3 +- common/components/hocs/Conditional.tsx | 25 ----------- common/components/hocs/index.ts | 1 - common/components/ui/ConditionalInput.tsx | 6 --- common/components/ui/Input.tsx | 44 ++++++++++++++++--- common/components/ui/SimpleSelect.tsx | 2 +- common/components/ui/TextArea.tsx | 44 ++++++++++++++++--- common/components/ui/index.ts | 1 - common/containers/Tabs/BroadcastTx/index.tsx | 2 +- .../components/TxHashInput.tsx | 5 ++- .../Tabs/Contracts/components/Deploy.tsx | 8 ++-- .../components/AmountField.tsx | 7 ++- .../components/InteractExplorer/index.tsx | 4 +- .../components/InteractForm/index.tsx | 12 ++--- .../Tabs/ENS/components/NameInput.tsx | 5 +-- .../components/Keystore/PaperWallet.tsx | 2 + .../components/Mnemonic/Word.tsx | 8 +++- .../Fields/ScheduleDepositField.tsx | 2 +- .../Fields/ScheduleGasLimitField.tsx | 2 +- .../Fields/ScheduleGasPriceField.tsx | 2 +- .../Fields/TimeBounty/TimeBountyField.tsx | 2 +- .../Fields/WindowSize/WindowSizeField.tsx | 2 +- .../Fields/WindowStart/WindowStartField.tsx | 2 +- .../components/SignMessage/index.tsx | 3 +- .../components/VerifyMessage/index.tsx | 3 +- .../Tabs/Swap/components/CurrencySwap.tsx | 14 +----- .../Tabs/Swap/components/LiteSend/Fields.tsx | 9 +++- .../Tabs/Swap/components/PaymentInfo.tsx | 1 + .../Tabs/Swap/components/ReceivingAddress.tsx | 3 +- .../Tabs/Swap/components/SupportFooter.tsx | 8 +++- common/libs/validators.ts | 8 ++-- common/libs/values.ts | 2 +- common/sagas/transaction/fields/fields.ts | 6 +-- common/sass/mixins.scss | 1 + 52 files changed, 196 insertions(+), 150 deletions(-) delete mode 100644 common/components/hocs/Conditional.tsx delete mode 100644 common/components/hocs/index.ts delete mode 100644 common/components/ui/ConditionalInput.tsx diff --git a/common/components/AddressField.tsx b/common/components/AddressField.tsx index 244bd713..7bd34e87 100644 --- a/common/components/AddressField.tsx +++ b/common/components/AddressField.tsx @@ -21,7 +21,7 @@ export const AddressField: React.SFC = ({ isReadOnly, isSelfAddress, isCh {translate(isSelfAddress ? 'X_ADDRESS' : 'SEND_ADDR')}
= ({