mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
Merge branch 'master' of github.com:MetaMask/metamask-extension into ci-screens
This commit is contained in:
commit
98e0fc1ab9
@ -232,7 +232,7 @@
|
|||||||
"done": {
|
"done": {
|
||||||
"message": "Fertig"
|
"message": "Fertig"
|
||||||
},
|
},
|
||||||
"downloadStatelogs": {
|
"downloadStateLogs": {
|
||||||
"message": "Statelogs herunterladen"
|
"message": "Statelogs herunterladen"
|
||||||
},
|
},
|
||||||
"dropped": {
|
"dropped": {
|
||||||
|
@ -247,7 +247,7 @@
|
|||||||
"done": {
|
"done": {
|
||||||
"message": "Completo"
|
"message": "Completo"
|
||||||
},
|
},
|
||||||
"downloadStatelogs": {
|
"downloadStateLogs": {
|
||||||
"message": "Descargar logs de estado"
|
"message": "Descargar logs de estado"
|
||||||
},
|
},
|
||||||
"dropped": {
|
"dropped": {
|
||||||
|
@ -223,7 +223,7 @@
|
|||||||
"done": {
|
"done": {
|
||||||
"message": "संपन्न"
|
"message": "संपन्न"
|
||||||
},
|
},
|
||||||
"downloadStatelogs": {
|
"downloadStateLogs": {
|
||||||
"message": "राज्य लॉग डाउनलोड करें"
|
"message": "राज्य लॉग डाउनलोड करें"
|
||||||
},
|
},
|
||||||
"edit": {
|
"edit": {
|
||||||
|
@ -223,7 +223,7 @@
|
|||||||
"done": {
|
"done": {
|
||||||
"message": "Gedaan"
|
"message": "Gedaan"
|
||||||
},
|
},
|
||||||
"downloadStatelogs": {
|
"downloadStateLogs": {
|
||||||
"message": "Staatslogboeken downloaden"
|
"message": "Staatslogboeken downloaden"
|
||||||
},
|
},
|
||||||
"edit": {
|
"edit": {
|
||||||
|
@ -3,13 +3,13 @@
|
|||||||
"message": "Принять"
|
"message": "Принять"
|
||||||
},
|
},
|
||||||
"account": {
|
"account": {
|
||||||
"message": "Аккаунт"
|
"message": "Счет"
|
||||||
},
|
},
|
||||||
"accountDetails": {
|
"accountDetails": {
|
||||||
"message": "Детали Аккаунта"
|
"message": "Детали счета"
|
||||||
},
|
},
|
||||||
"accountName": {
|
"accountName": {
|
||||||
"message": "Имя Пользователя"
|
"message": "Название счета"
|
||||||
},
|
},
|
||||||
"address": {
|
"address": {
|
||||||
"message": "Адрес"
|
"message": "Адрес"
|
||||||
@ -21,13 +21,13 @@
|
|||||||
"message": "Добавить токен"
|
"message": "Добавить токен"
|
||||||
},
|
},
|
||||||
"addTokens": {
|
"addTokens": {
|
||||||
"message": "Добавить Токены"
|
"message": "Добавить токены"
|
||||||
},
|
},
|
||||||
"amount": {
|
"amount": {
|
||||||
"message": "Количество"
|
"message": "Сумма"
|
||||||
},
|
},
|
||||||
"amountPlusGas": {
|
"amountPlusGas": {
|
||||||
"message": "Количество + газ"
|
"message": "Сумма + газ"
|
||||||
},
|
},
|
||||||
"appDescription": {
|
"appDescription": {
|
||||||
"message": "Расширение браузера для Ethereum",
|
"message": "Расширение браузера для Ethereum",
|
||||||
@ -37,11 +37,14 @@
|
|||||||
"message": "MetaMask",
|
"message": "MetaMask",
|
||||||
"description": "The name of the application"
|
"description": "The name of the application"
|
||||||
},
|
},
|
||||||
|
"approved": {
|
||||||
|
"message": "Одобрена"
|
||||||
|
},
|
||||||
"attemptingConnect": {
|
"attemptingConnect": {
|
||||||
"message": "Попытка подключиться к блокчейн сети."
|
"message": "Попытка подключиться к блокчейн сети."
|
||||||
},
|
},
|
||||||
"attributions": {
|
"attributions": {
|
||||||
"message": "Опознания"
|
"message": "Атрибуция"
|
||||||
},
|
},
|
||||||
"available": {
|
"available": {
|
||||||
"message": "Доступный"
|
"message": "Доступный"
|
||||||
@ -53,13 +56,13 @@
|
|||||||
"message": "Баланс:"
|
"message": "Баланс:"
|
||||||
},
|
},
|
||||||
"balances": {
|
"balances": {
|
||||||
"message": "Ваши балансы"
|
"message": "Ваш баланс"
|
||||||
},
|
},
|
||||||
"balanceIsInsufficientGas": {
|
"balanceIsInsufficientGas": {
|
||||||
"message": "Недостаточный баланс для текущего объема газа"
|
"message": "Недостаточный баланс для текущего объема газа"
|
||||||
},
|
},
|
||||||
"beta": {
|
"beta": {
|
||||||
"message": "БЕТА"
|
"message": "BETA"
|
||||||
},
|
},
|
||||||
"betweenMinAndMax": {
|
"betweenMinAndMax": {
|
||||||
"message": "должно быть больше или равно $1 и меньше или равно $2.",
|
"message": "должно быть больше или равно $1 и меньше или равно $2.",
|
||||||
@ -69,10 +72,10 @@
|
|||||||
"message": "Использовать Blockies Identicon"
|
"message": "Использовать Blockies Identicon"
|
||||||
},
|
},
|
||||||
"borrowDharma": {
|
"borrowDharma": {
|
||||||
"message": "Заимствовать с Dharma (бета)"
|
"message": "Взять в долг на Dharma (Beta)"
|
||||||
},
|
},
|
||||||
"builtInCalifornia": {
|
"builtInCalifornia": {
|
||||||
"message": "MetaMask спроектирован и построен в Калифорнии."
|
"message": "MetaMask спроектирован и разработан в Калифорнии."
|
||||||
},
|
},
|
||||||
"buy": {
|
"buy": {
|
||||||
"message": "Купить"
|
"message": "Купить"
|
||||||
@ -81,7 +84,10 @@
|
|||||||
"message": "Купить на Coinbase"
|
"message": "Купить на Coinbase"
|
||||||
},
|
},
|
||||||
"buyCoinbaseExplainer": {
|
"buyCoinbaseExplainer": {
|
||||||
"message": "Coinbase - самый популярный в мире способ купить и продать биткойн, ethereum и litecoin."
|
"message": "Биржа Coinbase – это наиболее популярный способ купить или продать bitcoin, ethereum и litecoin."
|
||||||
|
},
|
||||||
|
"ok": {
|
||||||
|
"message": "ОК"
|
||||||
},
|
},
|
||||||
"cancel": {
|
"cancel": {
|
||||||
"message": "Отмена"
|
"message": "Отмена"
|
||||||
@ -95,14 +101,17 @@
|
|||||||
"confirm": {
|
"confirm": {
|
||||||
"message": "Подтвердить"
|
"message": "Подтвердить"
|
||||||
},
|
},
|
||||||
|
"confirmed": {
|
||||||
|
"message": "Подтверждена"
|
||||||
|
},
|
||||||
"confirmContract": {
|
"confirmContract": {
|
||||||
"message": "Подтвердить Контракт"
|
"message": "Подтвердить контракт"
|
||||||
},
|
},
|
||||||
"confirmPassword": {
|
"confirmPassword": {
|
||||||
"message": "Подтвердите Пароль"
|
"message": "Подтвердите пароль"
|
||||||
},
|
},
|
||||||
"confirmTransaction": {
|
"confirmTransaction": {
|
||||||
"message": "Подтвердить Транзакцию"
|
"message": "Подтвердить транзакцию"
|
||||||
},
|
},
|
||||||
"continue": {
|
"continue": {
|
||||||
"message": "Продолжить"
|
"message": "Продолжить"
|
||||||
@ -114,7 +123,7 @@
|
|||||||
"message": "Развертывание контракта"
|
"message": "Развертывание контракта"
|
||||||
},
|
},
|
||||||
"conversionProgress": {
|
"conversionProgress": {
|
||||||
"message": "Выполняется конверсия"
|
"message": "Выполняется конвертация"
|
||||||
},
|
},
|
||||||
"copiedButton": {
|
"copiedButton": {
|
||||||
"message": "Скопировано"
|
"message": "Скопировано"
|
||||||
@ -126,7 +135,7 @@
|
|||||||
"message": "Скопировано!"
|
"message": "Скопировано!"
|
||||||
},
|
},
|
||||||
"copiedSafe": {
|
"copiedSafe": {
|
||||||
"message": "Я скопировал его где-то в безопасности"
|
"message": "Я скопировал это в безопасное место"
|
||||||
},
|
},
|
||||||
"copy": {
|
"copy": {
|
||||||
"message": "Скопировать"
|
"message": "Скопировать"
|
||||||
@ -138,29 +147,32 @@
|
|||||||
"message": " Скопировать "
|
"message": " Скопировать "
|
||||||
},
|
},
|
||||||
"copyPrivateKey": {
|
"copyPrivateKey": {
|
||||||
"message": "Это ваш личный ключ (нажмите, чтобы скопировать)"
|
"message": "Это ваш закрытый ключ (нажмите, чтобы скопировать)"
|
||||||
},
|
},
|
||||||
"create": {
|
"create": {
|
||||||
"message": "Создать"
|
"message": "Создать"
|
||||||
},
|
},
|
||||||
"createAccount": {
|
"createAccount": {
|
||||||
"message": "Регистрация"
|
"message": "Создать счет"
|
||||||
},
|
},
|
||||||
"createDen": {
|
"createDen": {
|
||||||
"message": "Создать"
|
"message": "Создать"
|
||||||
},
|
},
|
||||||
"crypto": {
|
"crypto": {
|
||||||
"message": "Крипто",
|
"message": "Криптовалюта",
|
||||||
"description": "Exchange type (cryptocurrencies)"
|
"description": "Exchange type (cryptocurrencies)"
|
||||||
},
|
},
|
||||||
"currentConversion": {
|
"currentConversion": {
|
||||||
"message": "Текущая конверсия"
|
"message": "Текущая конвертация"
|
||||||
},
|
},
|
||||||
"currentNetwork": {
|
"currentNetwork": {
|
||||||
"message": "Текущая сеть"
|
"message": "Текущая сеть"
|
||||||
},
|
},
|
||||||
"customGas": {
|
"customGas": {
|
||||||
"message": "Настроить Газ"
|
"message": "Настроить газ"
|
||||||
|
},
|
||||||
|
"customToken": {
|
||||||
|
"message": "Пользовательский токен"
|
||||||
},
|
},
|
||||||
"customize": {
|
"customize": {
|
||||||
"message": "Настроить"
|
"message": "Настроить"
|
||||||
@ -169,112 +181,114 @@
|
|||||||
"message": "Пользовательский RPC"
|
"message": "Пользовательский RPC"
|
||||||
},
|
},
|
||||||
"decimalsMustZerotoTen": {
|
"decimalsMustZerotoTen": {
|
||||||
"message": "Десятичные числа должны быть не менее 0, и не более 36."
|
"message": "Количество десятичных разрядов должно быть минимум 0 и максимум 36."
|
||||||
},
|
},
|
||||||
"decimal": {
|
"decimal": {
|
||||||
"message": "Десятичные значения точности"
|
"message": "Количество десятичных разрядов"
|
||||||
},
|
},
|
||||||
"defaultNetwork": {
|
"defaultNetwork": {
|
||||||
"message": "Сеть по умолчанию для транзакций Ether - это Main Net."
|
"message": "Основная сеть Ethereum – это сеть по умолчанию для Ether транзакций."
|
||||||
},
|
},
|
||||||
"denExplainer": {
|
"denExplainer": {
|
||||||
"message": "Ваш DEN - это ваше зашифрованное паролем хранилище в MetaMask."
|
"message": "DEN – это зашифрованное паролем хранилище внутри MetaMask."
|
||||||
},
|
},
|
||||||
"deposit": {
|
"deposit": {
|
||||||
"message": "Депозит"
|
"message": "Пополнить"
|
||||||
},
|
},
|
||||||
"depositBTC": {
|
"depositBTC": {
|
||||||
"message": "Депозит BTC по адресу:"
|
"message": "Отправьте ваш BTC на адрес ниже:"
|
||||||
},
|
},
|
||||||
"depositCoin": {
|
"depositCoin": {
|
||||||
"message": "Депозит $1 по указанному ниже адресу",
|
"message": "Отправьте ваш $1 на адрес ниже",
|
||||||
"description": "Tells the user what coin they have selected to deposit with shapeshift"
|
"description": "Tells the user what coin they have selected to deposit with shapeshift"
|
||||||
},
|
},
|
||||||
"depositEth": {
|
"depositEth": {
|
||||||
"message": "Депозит Eth"
|
"message": "Пополнить Eth"
|
||||||
},
|
},
|
||||||
"depositEther": {
|
"depositEther": {
|
||||||
"message": "Депозит Эфир"
|
"message": "Пополнить Ether"
|
||||||
},
|
},
|
||||||
"depositFiat": {
|
"depositFiat": {
|
||||||
"message": "Депозит с деньгами"
|
"message": "Пополнить деньгами"
|
||||||
},
|
},
|
||||||
"depositFromAccount": {
|
"depositFromAccount": {
|
||||||
"message": "Депозит с другого счета"
|
"message": "Пополнить с другого счета"
|
||||||
},
|
},
|
||||||
"depositShapeShift": {
|
"depositShapeShift": {
|
||||||
"message": "Депозит с ShapeShift"
|
"message": "Пополнить через ShapeShift"
|
||||||
},
|
},
|
||||||
"depositShapeShiftExplainer": {
|
"depositShapeShiftExplainer": {
|
||||||
"message": "Если у вас есть другие крипторесурсы, вы можете торговать и вносить Эфир непосредственно в кошелек MetaMask. Нет необходимости в аккаунте."
|
"message": "Если у вас есть другие криптовалюты, вы можете торговать и пополнять Ether напрямую в ваш MetaMask кошелек. Нет необходимости в счете."
|
||||||
},
|
},
|
||||||
"details": {
|
"details": {
|
||||||
"message": "Детали"
|
"message": "Детали"
|
||||||
},
|
},
|
||||||
"directDeposit": {
|
"directDeposit": {
|
||||||
"message": "Прямой Депозит"
|
"message": "Прямое пополнение"
|
||||||
},
|
},
|
||||||
"directDepositEther": {
|
"directDepositEther": {
|
||||||
"message": "Прямой Депозит Эфира"
|
"message": "Прямое пополнение Ether"
|
||||||
},
|
},
|
||||||
"directDepositEtherExplainer": {
|
"directDepositEtherExplainer": {
|
||||||
"message": "Если у вас уже есть Эфир, самый быстрый способ получить Эфир в вашем новом кошельке это прямым депозитом."
|
"message": "Если у вас уже есть Ether, то самый быстрый способ получить Ether в ваш новый кошелек – это прямое пополнение."
|
||||||
},
|
},
|
||||||
"done": {
|
"done": {
|
||||||
"message": "Готово"
|
"message": "Готово"
|
||||||
},
|
},
|
||||||
"downloadStatelogs": {
|
"downloadStateLogs": {
|
||||||
"message": "Загрузить логи статус"
|
"message": "Скачать журнал состояния"
|
||||||
|
"dropped": {
|
||||||
|
"message": "Отброшена"
|
||||||
},
|
},
|
||||||
"edit": {
|
"edit": {
|
||||||
"message": "Редактировать"
|
"message": "Редактировать"
|
||||||
},
|
},
|
||||||
"editAccountName": {
|
"editAccountName": {
|
||||||
"message": "Изменить Имя Аккаунта"
|
"message": "Редактировать название счета"
|
||||||
},
|
},
|
||||||
"emailUs": {
|
"emailUs": {
|
||||||
"message": "Свяжитесь с нами по электронной почте!"
|
"message": "Свяжитесь с нами по электронной почте!"
|
||||||
},
|
},
|
||||||
"encryptNewDen": {
|
"encryptNewDen": {
|
||||||
"message": "Шифруйте новый DEN"
|
"message": "Зашифровать ваш новый DEN"
|
||||||
},
|
},
|
||||||
"enterPassword": {
|
"enterPassword": {
|
||||||
"message": "Введите пароль"
|
"message": "Введите пароль"
|
||||||
},
|
},
|
||||||
"enterPasswordConfirm": {
|
"enterPasswordConfirm": {
|
||||||
"message": "Введите свой пароль для подтверждения"
|
"message": "Введите ваш пароль для подтверждения"
|
||||||
},
|
},
|
||||||
"etherscanView": {
|
"etherscanView": {
|
||||||
"message": "Просмотреть аккаунт на Etherscan"
|
"message": "Просмотреть счет на Etherscan"
|
||||||
},
|
},
|
||||||
"exchangeRate": {
|
"exchangeRate": {
|
||||||
"message": "Обменный Курс"
|
"message": "Обменный курс"
|
||||||
},
|
},
|
||||||
"exportPrivateKey": {
|
"exportPrivateKey": {
|
||||||
"message": "Экспорт закрытого ключа"
|
"message": "Экспортировать закрытый ключ"
|
||||||
},
|
},
|
||||||
"exportPrivateKeyWarning": {
|
"exportPrivateKeyWarning": {
|
||||||
"message": "Экспорт секретных ключей на свой страх и риск."
|
"message": "Вы экспортируете закрытые ключи на свой страх и риск."
|
||||||
},
|
},
|
||||||
"failed": {
|
"failed": {
|
||||||
"message": "Не смогли"
|
"message": "Неудачна"
|
||||||
},
|
},
|
||||||
"fiat": {
|
"fiat": {
|
||||||
"message": "Бумажные деньги",
|
"message": "Валюта",
|
||||||
"description": "Exchange type"
|
"description": "Exchange type"
|
||||||
},
|
},
|
||||||
"fileImportFail": {
|
"fileImportFail": {
|
||||||
"message": "Ошибка импорта файлов? Кликните сюда!",
|
"message": "Не работает импорт файла? Нажмите тут!",
|
||||||
"description": "Helps user import their account from a JSON file"
|
"description": "Helps user import their account from a JSON file"
|
||||||
},
|
},
|
||||||
"followTwitter": {
|
"followTwitter": {
|
||||||
"message": "Следуйте за нами на Twitter"
|
"message": "Читайте нас в Twitter"
|
||||||
},
|
},
|
||||||
"from": {
|
"from": {
|
||||||
"message": "Из"
|
"message": "Отправитель"
|
||||||
},
|
},
|
||||||
"fromToSame": {
|
"fromToSame": {
|
||||||
"message": "От и до адреса не могут быть одинаковым"
|
"message": "Адрес отправителя и получателя не могут быть одинаковыми"
|
||||||
},
|
},
|
||||||
"fromShapeShift": {
|
"fromShapeShift": {
|
||||||
"message": "Из ShapeShift"
|
"message": "Из ShapeShift"
|
||||||
@ -284,37 +298,37 @@
|
|||||||
"description": "Short indication of gas cost"
|
"description": "Short indication of gas cost"
|
||||||
},
|
},
|
||||||
"gasFee": {
|
"gasFee": {
|
||||||
"message": "Плата за Газ"
|
"message": "Комиссия за газ"
|
||||||
},
|
},
|
||||||
"gasLimit": {
|
"gasLimit": {
|
||||||
"message": "Газовый Предел"
|
"message": "Лимит газа"
|
||||||
},
|
},
|
||||||
"gasLimitCalculation": {
|
"gasLimitCalculation": {
|
||||||
"message": "Мы рассчитываем предполагаемый предел газа на основе коэффициентов успешности сети."
|
"message": "Мы расчитываем предлагаемый лимит газа на основании успешных ставок в сети."
|
||||||
},
|
},
|
||||||
"gasLimitRequired": {
|
"gasLimitRequired": {
|
||||||
"message": "Требуется ограничение на Газ"
|
"message": "Установите лимит газа"
|
||||||
},
|
},
|
||||||
"gasLimitTooLow": {
|
"gasLimitTooLow": {
|
||||||
"message": "Предел газа должен быть не менее 21000"
|
"message": "Лимит газа должен быть как минимум 21000"
|
||||||
},
|
},
|
||||||
"generatingSeed": {
|
"generatingSeed": {
|
||||||
"message": "Создание Семян ..."
|
"message": "Генерируем фразу..."
|
||||||
},
|
},
|
||||||
"gasPrice": {
|
"gasPrice": {
|
||||||
"message": "Цена на Газ (GWEI)"
|
"message": "Цена за газ (GWEI)"
|
||||||
},
|
},
|
||||||
"gasPriceCalculation": {
|
"gasPriceCalculation": {
|
||||||
"message": "Мы вычисляем предлагаемые цены на газ на основе коэффициентов успеха сети."
|
"message": "Мы расчитываем предлагаемые цены за газ на основании успешных ставок в сети."
|
||||||
},
|
},
|
||||||
"gasPriceRequired": {
|
"gasPriceRequired": {
|
||||||
"message": "Требуется цена на Газ"
|
"message": "Установите стоимость газа"
|
||||||
},
|
},
|
||||||
"getEther": {
|
"getEther": {
|
||||||
"message": "Получить Эфир"
|
"message": "Получить Ether"
|
||||||
},
|
},
|
||||||
"getEtherFromFaucet": {
|
"getEtherFromFaucet": {
|
||||||
"message": "Получите Эфир из крана $1",
|
"message": "Получить Ether из крана для $1",
|
||||||
"description": "Displays network name for Ether faucet"
|
"description": "Displays network name for Ether faucet"
|
||||||
},
|
},
|
||||||
"greaterThanMin": {
|
"greaterThanMin": {
|
||||||
@ -322,14 +336,14 @@
|
|||||||
"description": "helper for inputting hex as decimal input"
|
"description": "helper for inputting hex as decimal input"
|
||||||
},
|
},
|
||||||
"here": {
|
"here": {
|
||||||
"message": "здесь",
|
"message": "тут",
|
||||||
"description": "as in -click here- for more information (goes with troubleTokenBalances)"
|
"description": "as in -click here- for more information (goes with troubleTokenBalances)"
|
||||||
},
|
},
|
||||||
"hereList": {
|
"hereList": {
|
||||||
"message": "Вот список !!!!"
|
"message": "Вот список!!!!"
|
||||||
},
|
},
|
||||||
"hide": {
|
"hide": {
|
||||||
"message": "Спрятать"
|
"message": "Скрыть"
|
||||||
},
|
},
|
||||||
"hideToken": {
|
"hideToken": {
|
||||||
"message": "Скрыть токен"
|
"message": "Скрыть токен"
|
||||||
@ -338,33 +352,33 @@
|
|||||||
"message": "Скрыть токен?"
|
"message": "Скрыть токен?"
|
||||||
},
|
},
|
||||||
"howToDeposit": {
|
"howToDeposit": {
|
||||||
"message": "Как бы вы хотели поместить Эфир?"
|
"message": "Как бы вы хотели пополнить Ether?"
|
||||||
},
|
},
|
||||||
"holdEther": {
|
"holdEther": {
|
||||||
"message": "Это позволяет вам использовать эфир и токены и служит мостом для децентрализованных приложений."
|
"message": "Позволяет вам хранить ether и токены и служит в качестве моста в децентрализированные приложения."
|
||||||
},
|
},
|
||||||
"import": {
|
"import": {
|
||||||
"message": "Импортировать",
|
"message": "Импортировать",
|
||||||
"description": "Button to import an account from a selected file"
|
"description": "Button to import an account from a selected file"
|
||||||
},
|
},
|
||||||
"importAccount": {
|
"importAccount": {
|
||||||
"message": "Импорт Аккаунта"
|
"message": "Импортировать счет"
|
||||||
},
|
},
|
||||||
"importAccountMsg": {
|
"importAccountMsg": {
|
||||||
"message": " Импортированные аккаунты не будут связаны с вашей первоначально созданным аккаунтом MetaMask. Подробнее о импортированных аккаунтах "
|
"message":" Импортированные счета не будут ассоциированы с вашей ключевой фразой, созданной MetaMask. Узнать больше про импорт счетов "
|
||||||
},
|
},
|
||||||
"importAnAccount": {
|
"importAnAccount": {
|
||||||
"message": "Импортировать аккаунт"
|
"message": "Импортировать аккаунт"
|
||||||
},
|
},
|
||||||
"importDen": {
|
"importDen": {
|
||||||
"message": "Импорт существующих DEN"
|
"message": "Импортировать существующий DEN"
|
||||||
},
|
},
|
||||||
"imported": {
|
"imported": {
|
||||||
"message": "Импортирован",
|
"message": "Импортирован",
|
||||||
"description": "status showing that an account has been fully loaded into the keyring"
|
"description": "status showing that an account has been fully loaded into the keyring"
|
||||||
},
|
},
|
||||||
"infoHelp": {
|
"infoHelp": {
|
||||||
"message": "Информация и Помощь"
|
"message": "Информация и помощь"
|
||||||
},
|
},
|
||||||
"insufficientFunds": {
|
"insufficientFunds": {
|
||||||
"message": "Недостаточно средств."
|
"message": "Недостаточно средств."
|
||||||
@ -373,35 +387,44 @@
|
|||||||
"message": "Недостаточно токенов."
|
"message": "Недостаточно токенов."
|
||||||
},
|
},
|
||||||
"invalidAddress": {
|
"invalidAddress": {
|
||||||
"message": "Недействительный адрес"
|
"message": "Неверный адрес"
|
||||||
},
|
},
|
||||||
"invalidAddressRecipient": {
|
"invalidAddressRecipient": {
|
||||||
"message": "Недопустимый адрес получателя."
|
"message": "Неверный адрес получателя"
|
||||||
},
|
},
|
||||||
"invalidGasParams": {
|
"invalidGasParams": {
|
||||||
"message": "Недопустимые параметры Газа"
|
"message": "Неверные параметры газа"
|
||||||
},
|
},
|
||||||
"invalidInput": {
|
"invalidInput": {
|
||||||
"message": "Неправильный ввод."
|
"message": "Неверный ввод."
|
||||||
},
|
},
|
||||||
"invalidRequest": {
|
"invalidRequest": {
|
||||||
"message": "Неверный Запрос"
|
"message": "Неверный запрос"
|
||||||
},
|
},
|
||||||
"invalidRPC": {
|
"invalidRPC": {
|
||||||
"message": "Недопустимый URI RPC"
|
"message": "Неверный RPC URI"
|
||||||
},
|
},
|
||||||
"jsonFail": {
|
"jsonFail": {
|
||||||
"message": "Что-то пошло не так. Убедитесь, что ваш файл JSON правильно отформатирован."
|
"message": "Что-то пошло не так. Убедитесь, что ваш JSON файл правильно отформатирован."
|
||||||
},
|
},
|
||||||
"jsonFile": {
|
"jsonFile": {
|
||||||
"message": "Файл JSON",
|
"message": "JSON файл",
|
||||||
"description": "format for importing an account"
|
"description": "format for importing an account"
|
||||||
},
|
},
|
||||||
|
"keepTrackTokens": {
|
||||||
|
"message": "Следите за купленными вами токенами с помощью аккаунта MetaMask."
|
||||||
|
},
|
||||||
"kovan": {
|
"kovan": {
|
||||||
"message": "Kovan тестовая сеть"
|
"message": "Тестовая сеть Kovan"
|
||||||
},
|
},
|
||||||
"knowledgeDataBase": {
|
"knowledgeDataBase": {
|
||||||
"message": "Посетите нашу базу знаний"
|
"message": "Посмотрите нашу Базу Знаний"
|
||||||
|
},
|
||||||
|
"max": {
|
||||||
|
"message": "Максимум"
|
||||||
|
},
|
||||||
|
"learnMore": {
|
||||||
|
"message": "Узнать больше."
|
||||||
},
|
},
|
||||||
"lessThanMax": {
|
"lessThanMax": {
|
||||||
"message": "должно быть меньше или равно $1.",
|
"message": "должно быть меньше или равно $1.",
|
||||||
@ -410,29 +433,32 @@
|
|||||||
"likeToAddTokens": {
|
"likeToAddTokens": {
|
||||||
"message": "Вы хотите добавить эти токены?"
|
"message": "Вы хотите добавить эти токены?"
|
||||||
},
|
},
|
||||||
|
"links": {
|
||||||
|
"message": "Ссылки"
|
||||||
|
},
|
||||||
"limit": {
|
"limit": {
|
||||||
"message": "Предел"
|
"message": "Лимит"
|
||||||
},
|
},
|
||||||
"loading": {
|
"loading": {
|
||||||
"message": "Загрузка..."
|
"message": "Загрузка..."
|
||||||
},
|
},
|
||||||
"loadingTokens": {
|
"loadingTokens": {
|
||||||
"message": "Загрузка токенов ..."
|
"message": "Загрузка токенов..."
|
||||||
},
|
},
|
||||||
"localhost": {
|
"localhost": {
|
||||||
"message": "Локальный адрес 8545"
|
"message": "Localhost 8545"
|
||||||
},
|
},
|
||||||
"login": {
|
"login": {
|
||||||
"message": "Авторизоваться"
|
"message": "Вход"
|
||||||
},
|
},
|
||||||
"logout": {
|
"logout": {
|
||||||
"message": "Выйти"
|
"message": "Выход"
|
||||||
},
|
},
|
||||||
"loose": {
|
"loose": {
|
||||||
"message": "Рыхлый"
|
"message": "Несвязанный"
|
||||||
},
|
},
|
||||||
"loweCaseWords": {
|
"loweCaseWords": {
|
||||||
"message": "семенные слова имеют только символы нижнего регистра"
|
"message": "ключевая фраза может содержать только символы нижнего регистра"
|
||||||
},
|
},
|
||||||
"mainnet": {
|
"mainnet": {
|
||||||
"message": "Основная сеть Ethereum"
|
"message": "Основная сеть Ethereum"
|
||||||
@ -441,19 +467,19 @@
|
|||||||
"message": "Сообщение"
|
"message": "Сообщение"
|
||||||
},
|
},
|
||||||
"metamaskDescription": {
|
"metamaskDescription": {
|
||||||
"message": "MetaMask - это безопасное хранилище для Ethereum."
|
"message": "MetaMask – безопасный кошелек для Ethereum."
|
||||||
},
|
},
|
||||||
"min": {
|
"min": {
|
||||||
"message": "Минимум"
|
"message": "Минимум"
|
||||||
},
|
},
|
||||||
"myAccounts": {
|
"myAccounts": {
|
||||||
"message": "Мои Аккаунты"
|
"message": "Мои счета"
|
||||||
},
|
},
|
||||||
"mustSelectOne": {
|
"mustSelectOne": {
|
||||||
"message": "Необходимо выбрать не менее 1 токена."
|
"message": "Необходимо выбрать как минимум 1 токен."
|
||||||
},
|
},
|
||||||
"needEtherInWallet": {
|
"needEtherInWallet": {
|
||||||
"message": "Чтобы взаимодействовать с децентрализованными приложениями с помощью MetaMask, вам понадобится Эфир в вашем кошельке."
|
"message": "Для взаимодействия с децентрализованными приложениями с помощью MetaMask нужен Ether в вашем кошельке."
|
||||||
},
|
},
|
||||||
"needImportFile": {
|
"needImportFile": {
|
||||||
"message": "Вы должны выбрать файл для импорта.",
|
"message": "Вы должны выбрать файл для импорта.",
|
||||||
@ -464,60 +490,60 @@
|
|||||||
"description": "Password and file needed to import an account"
|
"description": "Password and file needed to import an account"
|
||||||
},
|
},
|
||||||
"negativeETH": {
|
"negativeETH": {
|
||||||
"message": "Невозможно отправить отрицательные количества ETH."
|
"message": "Невозможно отправить отрицательную сумму ETH."
|
||||||
},
|
},
|
||||||
"networks": {
|
"networks": {
|
||||||
"message": "Сети"
|
"message": "Сети"
|
||||||
},
|
},
|
||||||
"newAccount": {
|
"newAccount": {
|
||||||
"message": "Новый Аккаунт"
|
"message": "Новый счет"
|
||||||
},
|
},
|
||||||
"newAccountNumberName": {
|
"newAccountNumberName": {
|
||||||
"message": "Аккаунт $1",
|
"message": "Счет $1",
|
||||||
"description": "Default name of next account to be created on create account screen"
|
"description": "Default name of next account to be created on create account screen"
|
||||||
},
|
},
|
||||||
"newContract": {
|
"newContract": {
|
||||||
"message": "Новый Контракт"
|
"message": "Новый контракт"
|
||||||
},
|
},
|
||||||
"newPassword": {
|
"newPassword": {
|
||||||
"message": "Новый пароль (мин. 8 символов)"
|
"message": "Новый пароль (мин. 8 символов)"
|
||||||
},
|
},
|
||||||
"newRecipient": {
|
"newRecipient": {
|
||||||
"message": "Новый Получатель"
|
"message": "Новый получатель"
|
||||||
},
|
},
|
||||||
"newRPC": {
|
"newRPC": {
|
||||||
"message": "Новый URL-адрес RPC"
|
"message": "Новый RPC URL"
|
||||||
},
|
},
|
||||||
"next": {
|
"next": {
|
||||||
"message": "Далее"
|
"message": "Далее"
|
||||||
},
|
},
|
||||||
"noAddressForName": {
|
"noAddressForName": {
|
||||||
"message": "Для этого имени не задан адрес."
|
"message": "Дла этого названия не установлен адрес."
|
||||||
},
|
},
|
||||||
"noDeposits": {
|
"noDeposits": {
|
||||||
"message": "Не было получено никаких депозитов"
|
"message": "Пополнения не получены"
|
||||||
},
|
},
|
||||||
"noTransactionHistory": {
|
"noTransactionHistory": {
|
||||||
"message": "Нет истории транзакций."
|
"message": "Нет истории транзакций."
|
||||||
},
|
},
|
||||||
"noTransactions": {
|
"noTransactions": {
|
||||||
"message": "Нет Транзакций"
|
"message": "Нет транзакций"
|
||||||
},
|
},
|
||||||
"notStarted": {
|
"notStarted": {
|
||||||
"message": "Не Начался"
|
"message": "Не запущен"
|
||||||
},
|
},
|
||||||
"oldUI": {
|
"oldUI": {
|
||||||
"message": "Старый Интерфейс"
|
"message": "Старая версия интерфейса"
|
||||||
},
|
},
|
||||||
"oldUIMessage": {
|
"oldUIMessage": {
|
||||||
"message": "Вы вернулись к старому интерфейсу. Вы можете вернуться к новому с помощью опции в раскрывающемся меню в правом верхнем углу."
|
"message": "Вы вернулись к старой версии интерфейса пользователя. Вы можете переключиться на новую с помощью опции выпадающего меню в правом верхнем углу."
|
||||||
},
|
},
|
||||||
"or": {
|
"or": {
|
||||||
"message": "или",
|
"message": "или",
|
||||||
"description": "choice between creating or importing a new account"
|
"description": "choice between creating or importing a new account"
|
||||||
},
|
},
|
||||||
"passwordCorrect": {
|
"passwordCorrect": {
|
||||||
"message": "Убедитесь, что ваш пароль правильный."
|
"message": "Убедитесь, что ваш пароль верный."
|
||||||
},
|
},
|
||||||
"passwordMismatch": {
|
"passwordMismatch": {
|
||||||
"message": "пароли не совпадают",
|
"message": "пароли не совпадают",
|
||||||
@ -528,27 +554,30 @@
|
|||||||
"description": "in password creation process, the password is not long enough to be secure"
|
"description": "in password creation process, the password is not long enough to be secure"
|
||||||
},
|
},
|
||||||
"pastePrivateKey": {
|
"pastePrivateKey": {
|
||||||
"message": "Вставьте свою личную строку:",
|
"message": "Вставьте ваш закрытый ключ тут:",
|
||||||
"description": "For importing an account from a private key"
|
"description": "For importing an account from a private key"
|
||||||
},
|
},
|
||||||
"pasteSeed": {
|
"pasteSeed": {
|
||||||
"message": "Вставьте здесь свою семенную фразу!"
|
"message": "Вставьте вашу ключевую фразу!"
|
||||||
},
|
},
|
||||||
"personalAddressDetected": {
|
"personalAddressDetected": {
|
||||||
"message": "Персональный адрес обнаружен. Введите адрес контракта токена."
|
"message": "Обнаружен персональный адрес. Введите адрес контракта токена."
|
||||||
},
|
},
|
||||||
"pleaseReviewTransaction": {
|
"pleaseReviewTransaction": {
|
||||||
"message": "Проверьте транзакцию."
|
"message": "Проверьте транзакцию."
|
||||||
},
|
},
|
||||||
|
"popularTokens": {
|
||||||
|
"message": "Популярные токены"
|
||||||
|
},
|
||||||
"privacyMsg": {
|
"privacyMsg": {
|
||||||
"message": "Политика Конфиденциальности"
|
"message": "Политика конфиденциальности"
|
||||||
},
|
},
|
||||||
"privateKey": {
|
"privateKey": {
|
||||||
"message": "Закрытый ключ",
|
"message": "Закрытый ключ",
|
||||||
"description": "select this type of file to use to import an account"
|
"description": "select this type of file to use to import an account"
|
||||||
},
|
},
|
||||||
"privateKeyWarning": {
|
"privateKeyWarning": {
|
||||||
"message": "Предупреждение: никогда не раскрывайте этот ключ. Любой, у кого есть ваши личные ключи, может украсть любые активы, хранящиеся в вашем аккаунте."
|
"message": "Предупреждение: Никогда не раскрывайте этот ключ. Любой, у кого есть ваши закрытые ключи, может украсть любые активы, хранящиеся на счету."
|
||||||
},
|
},
|
||||||
"privateNetwork": {
|
"privateNetwork": {
|
||||||
"message": "Частная сеть"
|
"message": "Частная сеть"
|
||||||
@ -557,126 +586,165 @@
|
|||||||
"message": "Показать QR-код"
|
"message": "Показать QR-код"
|
||||||
},
|
},
|
||||||
"readdToken": {
|
"readdToken": {
|
||||||
"message": "Вы можете добавить этот токен в будущем, перейдя в “Добавить токен” в меню параметров вашего аккаунта."
|
"message": "Вы можете в будущем добавить обратно этот токен, выбрав пункт меню “Добавить токен”."
|
||||||
},
|
},
|
||||||
"readMore": {
|
"readMore": {
|
||||||
"message": "Подробнее читайте здесь."
|
"message": "Узнать больше тут."
|
||||||
},
|
},
|
||||||
"readMore2": {
|
"readMore2": {
|
||||||
"message": "Прочитайте больше."
|
"message": "Узнать больше."
|
||||||
},
|
},
|
||||||
"receive": {
|
"receive": {
|
||||||
"message": "Получить"
|
"message": "Получить"
|
||||||
},
|
},
|
||||||
"recipientAddress": {
|
"recipientAddress": {
|
||||||
"message": "Адрес Получателя"
|
"message": "Адрес получателя"
|
||||||
},
|
},
|
||||||
"refundAddress": {
|
"refundAddress": {
|
||||||
"message": "Ваш Адрес Возврата"
|
"message": "Ваш адрес для возврата средств"
|
||||||
},
|
},
|
||||||
"rejected": {
|
"rejected": {
|
||||||
"message": "Отклонено"
|
"message": "Отклонена"
|
||||||
},
|
},
|
||||||
"resetAccount": {
|
"resetAccount": {
|
||||||
"message": "Сбросить аккаунт"
|
"message": "Сбросить аккаунт"
|
||||||
},
|
},
|
||||||
"restoreFromSeed": {
|
"restoreFromSeed": {
|
||||||
"message": "Восстановить от семенной фразы"
|
"message": "Восстановить из ключевой фразы"
|
||||||
|
},
|
||||||
|
"restoreVault": {
|
||||||
|
"message": "Восстановить кошелек"
|
||||||
},
|
},
|
||||||
"required": {
|
"required": {
|
||||||
"message": "Необходимо"
|
"message": "Обязательное поле"
|
||||||
},
|
},
|
||||||
"retryWithMoreGas": {
|
"retryWithMoreGas": {
|
||||||
"message": "Повторите попытку с более высокой ценой на газ здесь"
|
"message": "Повторите попытку с большей ценой за газRetry with a higher gas price here"
|
||||||
|
},
|
||||||
|
"walletSeed": {
|
||||||
|
"message": "Ключевая фраза кошелька"
|
||||||
},
|
},
|
||||||
"revealSeedWords": {
|
"revealSeedWords": {
|
||||||
"message": "Раскрыть семенные слова"
|
"message": "Показать ключевую фразу"
|
||||||
},
|
},
|
||||||
"revealSeedWordsWarning": {
|
"revealSeedWordsWarning": {
|
||||||
"message": "Не восстанавливайте семенные слова в общественном месте! Эти слова могут использоваться для кражи всех ваших аккаунтах."
|
"message": "Не восстанавливайте ключевую фразу в общественном месте! Она может быть использована для кражи всех ваших счетов."
|
||||||
},
|
},
|
||||||
"revert": {
|
"revert": {
|
||||||
"message": "Откат"
|
"message": "Восстановить"
|
||||||
},
|
},
|
||||||
"rinkeby": {
|
"rinkeby": {
|
||||||
"message": "Rinkeby тестовая сеть"
|
"message": "Тестовая сеть Rinkeby"
|
||||||
},
|
},
|
||||||
"ropsten": {
|
"ropsten": {
|
||||||
"message": "Ropsten тестовая сеть"
|
"message": "Тестовая сеть Ropsten"
|
||||||
|
},
|
||||||
|
"currentRpc": {
|
||||||
|
"message": "Current RPC"
|
||||||
|
},
|
||||||
|
"connectingToMainnet": {
|
||||||
|
"message": "Соединение с основной сетью Ethereum"
|
||||||
|
},
|
||||||
|
"connectingToRopsten": {
|
||||||
|
"message": "Соединение с тестовой сетью Ropsten"
|
||||||
|
},
|
||||||
|
"connectingToKovan": {
|
||||||
|
"message": "Соединение с тестовой сетью Kovan"
|
||||||
|
},
|
||||||
|
"connectingToRinkeby": {
|
||||||
|
"message": "Соединение с тестовой сетью Rinkeby"
|
||||||
|
},
|
||||||
|
"connectingToUnknown": {
|
||||||
|
"message": "Соединение с неизвестной сетью"
|
||||||
},
|
},
|
||||||
"sampleAccountName": {
|
"sampleAccountName": {
|
||||||
"message": "Например, Мой новый аккаунт",
|
"message": "Например, Мой новый счет",
|
||||||
"description": "Help user understand concept of adding a human-readable name to their account"
|
"description": "Help user understand concept of adding a human-readable name to their account"
|
||||||
},
|
},
|
||||||
"save": {
|
"save": {
|
||||||
"message": "Сохранить"
|
"message": "Сохранить"
|
||||||
},
|
},
|
||||||
"saveAsFile": {
|
"saveAsFile": {
|
||||||
"message": "Сохранить как Файл",
|
"message": "Сохранить в виде файла",
|
||||||
"description": "Account export process"
|
"description": "Account export process"
|
||||||
},
|
},
|
||||||
"saveSeedAsFile": {
|
"saveSeedAsFile": {
|
||||||
"message": "Сохранить Семенные Слова Как Файл"
|
"message": "Сохранить ключевую фразу в виде файла"
|
||||||
},
|
},
|
||||||
"search": {
|
"search": {
|
||||||
"message": "Поиск"
|
"message": "Поиск"
|
||||||
},
|
},
|
||||||
"secretPhrase": {
|
"secretPhrase": {
|
||||||
"message": "Введите свою секретную двенадцатисловную фразу здесь, чтобы восстановить хранилище."
|
"message": "Введите вашу ключевую фразу из 12 слов, чтобы восстановить кошелек."
|
||||||
|
},
|
||||||
|
"newPassword8Chars": {
|
||||||
|
"message": "Новый пароль (мин. 8 символов)"
|
||||||
},
|
},
|
||||||
"seedPhraseReq": {
|
"seedPhraseReq": {
|
||||||
"message": "семенные фразы длиной 12 слов"
|
"message": "ключевые фразы имеют длину 12 слов"
|
||||||
},
|
},
|
||||||
"select": {
|
"select": {
|
||||||
"message": "Выбрать"
|
"message": "Выбрать"
|
||||||
},
|
},
|
||||||
"selectCurrency": {
|
"selectCurrency": {
|
||||||
"message": "Выберите Валюту"
|
"message": "Выберите валюту"
|
||||||
},
|
},
|
||||||
"selectService": {
|
"selectService": {
|
||||||
"message": "Выберите Сервис"
|
"message": "Выберите сервис"
|
||||||
},
|
},
|
||||||
"selectType": {
|
"selectType": {
|
||||||
"message": "Выберите Тип"
|
"message": "Выберите тип"
|
||||||
},
|
},
|
||||||
"send": {
|
"send": {
|
||||||
"message": "Послать"
|
"message": "Отправить"
|
||||||
},
|
},
|
||||||
"sendETH": {
|
"sendETH": {
|
||||||
"message": "Отправить ETH"
|
"message": "Отправить ETH"
|
||||||
},
|
},
|
||||||
"sendTokens": {
|
"sendTokens": {
|
||||||
"message": "Отправить Токены"
|
"message": "Отправить токены"
|
||||||
|
},
|
||||||
|
"onlySendToEtherAddress": {
|
||||||
|
"message": "Отправляйте ETH только на Ethereum адреса."
|
||||||
|
},
|
||||||
|
"searchTokens": {
|
||||||
|
"message": "Поиск токенов"
|
||||||
},
|
},
|
||||||
"sendTokensAnywhere": {
|
"sendTokensAnywhere": {
|
||||||
"message": "Отправить Токены кому-либо с аккаунтом Ethereum"
|
"message": "Отправить токены любому, у кого есть счет Ethereum"
|
||||||
},
|
},
|
||||||
"settings": {
|
"settings": {
|
||||||
"message": "Настройки"
|
"message": "Настройки"
|
||||||
},
|
},
|
||||||
|
"info": {
|
||||||
|
"message": "Информация"
|
||||||
|
},
|
||||||
"shapeshiftBuy": {
|
"shapeshiftBuy": {
|
||||||
"message": "Купить с помощью Shapeshift"
|
"message": "Купить через Shapeshift"
|
||||||
},
|
},
|
||||||
"showPrivateKeys": {
|
"showPrivateKeys": {
|
||||||
"message": "Показать приватные ключи"
|
"message": "Показать закрытые ключи"
|
||||||
},
|
},
|
||||||
"showQRCode": {
|
"showQRCode": {
|
||||||
"message": "Показать QR-код"
|
"message": "Показать QR-код"
|
||||||
},
|
},
|
||||||
"sign": {
|
"sign": {
|
||||||
"message": "Знак"
|
"message": "Подпись"
|
||||||
|
},
|
||||||
|
"signed": {
|
||||||
|
"message": "Подписана"
|
||||||
},
|
},
|
||||||
"signMessage": {
|
"signMessage": {
|
||||||
"message": "Нодписать сообщение"
|
"message": "Подписать сообщение"
|
||||||
},
|
},
|
||||||
"signNotice": {
|
"signNotice": {
|
||||||
"message": "Подписание этого сообщения может иметь \nопасные побочные эффекты. Только подписывайте сообщения \nс сайтов, которым вы полностью доверяете своим аккаунтом. Этот опасный метод будет удален в будущей версии."
|
"message": "Подпись этого сообщения может иметь \nопасные побочные эффекты. Подписывайте только сообщения \nс сайтов, которым вы полностью доверяете свой аккаунт. Этот опасный метод будет удален в будущей версии."
|
||||||
},
|
},
|
||||||
"sigRequest": {
|
"sigRequest": {
|
||||||
"message": "Запрос на подпись"
|
"message": "Запрос подписи"
|
||||||
},
|
},
|
||||||
"sigRequested": {
|
"sigRequested": {
|
||||||
"message": "Подпись Запрошена"
|
"message": "Подпись запрошена"
|
||||||
},
|
},
|
||||||
"spaceBetween": {
|
"spaceBetween": {
|
||||||
"message": "между словами может быть только пробел"
|
"message": "между словами может быть только пробел"
|
||||||
@ -685,53 +753,59 @@
|
|||||||
"message": "Статус"
|
"message": "Статус"
|
||||||
},
|
},
|
||||||
"stateLogs": {
|
"stateLogs": {
|
||||||
"message": "Логи Статуса"
|
"message": "Журнал состояния"
|
||||||
},
|
},
|
||||||
"stateLogsDescription": {
|
"stateLogsDescription": {
|
||||||
"message": "Логи статуса содержат ваши общедоступные адреса и отправленные транзакции."
|
"message": "Журнал состояния содержит ваши публичные адреса счетов и совершенные транзакции."
|
||||||
|
},
|
||||||
|
"stateLogError": {
|
||||||
|
"message": "Ошибка при получении журнала состояния."
|
||||||
},
|
},
|
||||||
"submit": {
|
"submit": {
|
||||||
"message": "Отправить"
|
"message": "Отправить"
|
||||||
},
|
},
|
||||||
|
"submitted": {
|
||||||
|
"message": "Отправлена"
|
||||||
|
},
|
||||||
"supportCenter": {
|
"supportCenter": {
|
||||||
"message": "Посетите наш Центр поддержки"
|
"message": "Перейти в наш Центр поддержки"
|
||||||
},
|
},
|
||||||
"symbolBetweenZeroTen": {
|
"symbolBetweenZeroTen": {
|
||||||
"message": "Символ должен быть от 0 до 10 символов."
|
"message": "Символ должен быть от 0 до 10 символов."
|
||||||
},
|
},
|
||||||
"takesTooLong": {
|
"takesTooLong": {
|
||||||
"message": "Занимает слишком долго?"
|
"message": "Слишком долго?"
|
||||||
},
|
},
|
||||||
"terms": {
|
"terms": {
|
||||||
"message": "Условия Эксплуатации"
|
"message": "Условия пользования"
|
||||||
},
|
},
|
||||||
"testFaucet": {
|
"testFaucet": {
|
||||||
"message": "Тестовый Кран"
|
"message": "Тестовый кран"
|
||||||
},
|
},
|
||||||
"to": {
|
"to": {
|
||||||
"message": "К"
|
"message": "Получатель: "
|
||||||
},
|
},
|
||||||
"toETHviaShapeShift": {
|
"toETHviaShapeShift": {
|
||||||
"message": "$1 в ETH через ShapeShift",
|
"message": "$1 в ETH через ShapeShift",
|
||||||
"description": "system will fill in deposit type in start of message"
|
"description": "system will fill in deposit type in start of message"
|
||||||
},
|
},
|
||||||
"tokenAddress": {
|
"tokenAddress": {
|
||||||
"message": "Адрес Токена"
|
"message": "Адрес токена"
|
||||||
},
|
},
|
||||||
"tokenAlreadyAdded": {
|
"tokenAlreadyAdded": {
|
||||||
"message": "Токен уже добавлен."
|
"message": "Токен уже был добавлен."
|
||||||
},
|
},
|
||||||
"tokenBalance": {
|
"tokenBalance": {
|
||||||
"message": "Баланс Вашых Tокенов:"
|
"message": "Баланс ваших токенов:"
|
||||||
},
|
},
|
||||||
"tokenSelection": {
|
"tokenSelection": {
|
||||||
"message": "Поиск токенов или выбор из нашего списка популярных токенов."
|
"message": "Поищите токен или выберите из нашего списка популярных токенов."
|
||||||
},
|
},
|
||||||
"tokenSymbol": {
|
"tokenSymbol": {
|
||||||
"message": "Символ Токена"
|
"message": "Символ токена"
|
||||||
},
|
},
|
||||||
"tokenWarning1": {
|
"tokenWarning1": {
|
||||||
"message": "Следите за токенами, которые вы купили с помощью аккаунта MetaMask. Если вы купили токены, используя другой аккаунт, эти токены здесь не появятся."
|
"message": "Отслеживаются токены, купленные на счет в MetaMask. Если вы купили токены, используя другой счет, такие токены не будут тут отображены."
|
||||||
},
|
},
|
||||||
"total": {
|
"total": {
|
||||||
"message": "Всего"
|
"message": "Всего"
|
||||||
@ -740,35 +814,38 @@
|
|||||||
"message": "транзакции"
|
"message": "транзакции"
|
||||||
},
|
},
|
||||||
"transactionMemo": {
|
"transactionMemo": {
|
||||||
"message": "Транзакционная записка (необязательно)"
|
"message": "Транзакционные данные (необязательный)"
|
||||||
},
|
},
|
||||||
"transactionNumber": {
|
"transactionNumber": {
|
||||||
"message": "Номер Транзакции"
|
"message": "Номер транзакции"
|
||||||
},
|
},
|
||||||
"transfers": {
|
"transfers": {
|
||||||
"message": "Переводы"
|
"message": "Переводы"
|
||||||
},
|
},
|
||||||
"troubleTokenBalances": {
|
"troubleTokenBalances": {
|
||||||
"message": "У нас были проблемы с загрузкой ваших токенов. Вы можете просмотреть их ",
|
"message": "Возникли проблемы при загрузке балансов токенов. Вы можете посмотреть их ",
|
||||||
"description": "Followed by a link (here) to view token balances"
|
"description": "Followed by a link (here) to view token balances"
|
||||||
},
|
},
|
||||||
"twelveWords": {
|
"twelveWords": {
|
||||||
"message": "Эти 12 слов - единственный способ восстановить ваши учетные записи MetaMask.\nСохраните их где-нибудь в безопасности и в тайне."
|
"message": "Эти 12 слов являются единственной возможностью восстановить ваши счета в MetaMask.\nСохраните из в надежном секретном месте."
|
||||||
},
|
},
|
||||||
"typePassword": {
|
"typePassword": {
|
||||||
"message": "Введите Пароль"
|
"message": "Введите пароль"
|
||||||
},
|
},
|
||||||
"uiWelcome": {
|
"uiWelcome": {
|
||||||
"message": "Добро пожаловать в новый интерфейс (бета-версия)"
|
"message": "Новый интерфейс (Beta)"
|
||||||
},
|
},
|
||||||
"uiWelcomeMessage": {
|
"uiWelcomeMessage": {
|
||||||
"message": "Теперь вы используете новый интерфейс Metamask. Осмотритесь, попробуйте новые функции, такие как отправку токенов, и сообщите нам, есть ли у вас какие-либо проблемы."
|
"message": "Теперь вы используете новый интерфейс пользователя MetaMask. Осмотритесь, попробуйте новые функции, например, отправить токены и, если возникнут проблемы, сообщите нам."
|
||||||
|
},
|
||||||
|
"unapproved": {
|
||||||
|
"message": "Не одобрена"
|
||||||
},
|
},
|
||||||
"unavailable": {
|
"unavailable": {
|
||||||
"message": "Недоступен"
|
"message": "Недоступный"
|
||||||
},
|
},
|
||||||
"unknown": {
|
"unknown": {
|
||||||
"message": "Неизвестный"
|
"message": "Неизвестно"
|
||||||
},
|
},
|
||||||
"unknownNetwork": {
|
"unknownNetwork": {
|
||||||
"message": "Неизвестная частная сеть"
|
"message": "Неизвестная частная сеть"
|
||||||
@ -777,7 +854,7 @@
|
|||||||
"message": "Неизвестный идентификатор сети"
|
"message": "Неизвестный идентификатор сети"
|
||||||
},
|
},
|
||||||
"uriErrorMsg": {
|
"uriErrorMsg": {
|
||||||
"message": "Для URI требуется соответствующий префикс HTTP / HTTPS."
|
"message": "Для URI требуется соответствующий префикс HTTP/HTTPS."
|
||||||
},
|
},
|
||||||
"usaOnly": {
|
"usaOnly": {
|
||||||
"message": "Только США",
|
"message": "Только США",
|
||||||
@ -787,19 +864,19 @@
|
|||||||
"message": "Используется различными клиентами"
|
"message": "Используется различными клиентами"
|
||||||
},
|
},
|
||||||
"useOldUI": {
|
"useOldUI": {
|
||||||
"message": "Использовать старый интерфейс"
|
"message": "Использовать старый интерфейс пользователя"
|
||||||
},
|
},
|
||||||
"validFileImport": {
|
"validFileImport": {
|
||||||
"message": "Вы должны выбрать действительный файл для импорта."
|
"message": "Вам нужно выбрать правильный файл для импорта."
|
||||||
},
|
},
|
||||||
"vaultCreated": {
|
"vaultCreated": {
|
||||||
"message": "Создано хранилище"
|
"message": "Кошелек был создан"
|
||||||
},
|
},
|
||||||
"viewAccount": {
|
"viewAccount": {
|
||||||
"message": "Посмотреть аккаунт"
|
"message": "Посмотреть счет"
|
||||||
},
|
},
|
||||||
"visitWebSite": {
|
"visitWebSite": {
|
||||||
"message": "Посетите наш сайт"
|
"message": "Перейти на наш сайт"
|
||||||
},
|
},
|
||||||
"warning": {
|
"warning": {
|
||||||
"message": "Предупреждение"
|
"message": "Предупреждение"
|
||||||
@ -811,7 +888,7 @@
|
|||||||
"message": "Что это?"
|
"message": "Что это?"
|
||||||
},
|
},
|
||||||
"yourSigRequested": {
|
"yourSigRequested": {
|
||||||
"message": "Ваша подпись запрашивается"
|
"message": "Запрашивается ваша подпись"
|
||||||
},
|
},
|
||||||
"youSign": {
|
"youSign": {
|
||||||
"message": "Вы подписываете"
|
"message": "Вы подписываете"
|
||||||
|
@ -223,7 +223,7 @@
|
|||||||
"done": {
|
"done": {
|
||||||
"message": "Končano"
|
"message": "Končano"
|
||||||
},
|
},
|
||||||
"downloadStatelogs": {
|
"downloadStateLogs": {
|
||||||
"message": "Prenesi state dnevnike"
|
"message": "Prenesi state dnevnike"
|
||||||
},
|
},
|
||||||
"edit": {
|
"edit": {
|
||||||
|
@ -223,7 +223,7 @@
|
|||||||
"done": {
|
"done": {
|
||||||
"message": "เสร็จสิ้น"
|
"message": "เสร็จสิ้น"
|
||||||
},
|
},
|
||||||
"downloadStatelogs": {
|
"downloadStateLogs": {
|
||||||
"message": "ดาวน์โหลดล็อกสถานะ"
|
"message": "ดาวน์โหลดล็อกสถานะ"
|
||||||
},
|
},
|
||||||
"edit": {
|
"edit": {
|
||||||
|
@ -235,7 +235,7 @@
|
|||||||
"done": {
|
"done": {
|
||||||
"message": "完成"
|
"message": "完成"
|
||||||
},
|
},
|
||||||
"downloadStatelogs": {
|
"downloadStateLogs": {
|
||||||
"message": "下載狀態紀錄"
|
"message": "下載狀態紀錄"
|
||||||
},
|
},
|
||||||
"dropped": {
|
"dropped": {
|
||||||
|
@ -161,9 +161,11 @@ module.exports = class TransactionController extends EventEmitter {
|
|||||||
this.emit(`${txMeta.id}:unapproved`, txMeta)
|
this.emit(`${txMeta.id}:unapproved`, txMeta)
|
||||||
}
|
}
|
||||||
|
|
||||||
async newUnapprovedTransaction (txParams) {
|
async newUnapprovedTransaction (txParams, opts = {}) {
|
||||||
log.debug(`MetaMaskController newUnapprovedTransaction ${JSON.stringify(txParams)}`)
|
log.debug(`MetaMaskController newUnapprovedTransaction ${JSON.stringify(txParams)}`)
|
||||||
const initialTxMeta = await this.addUnapprovedTransaction(txParams)
|
const initialTxMeta = await this.addUnapprovedTransaction(txParams)
|
||||||
|
initialTxMeta.origin = opts.origin
|
||||||
|
this.txStateManager.updateTx(initialTxMeta, '#newUnapprovedTransaction - adding the origin')
|
||||||
// listen for tx completion (success, fail)
|
// listen for tx completion (success, fail)
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
this.txStateManager.once(`${initialTxMeta.id}:finished`, (finishedTxMeta) => {
|
this.txStateManager.once(`${initialTxMeta.id}:finished`, (finishedTxMeta) => {
|
||||||
|
@ -57,7 +57,6 @@ module.exports = class MetamaskController extends EventEmitter {
|
|||||||
this.defaultMaxListeners = 20
|
this.defaultMaxListeners = 20
|
||||||
|
|
||||||
this.sendUpdate = debounce(this.privateSendUpdate.bind(this), 200)
|
this.sendUpdate = debounce(this.privateSendUpdate.bind(this), 200)
|
||||||
|
|
||||||
this.opts = opts
|
this.opts = opts
|
||||||
const initState = opts.initState || {}
|
const initState = opts.initState || {}
|
||||||
this.recordFirstTimeInfo(initState)
|
this.recordFirstTimeInfo(initState)
|
||||||
@ -242,6 +241,11 @@ module.exports = class MetamaskController extends EventEmitter {
|
|||||||
static: {
|
static: {
|
||||||
eth_syncing: false,
|
eth_syncing: false,
|
||||||
web3_clientVersion: `MetaMask/v${version}`,
|
web3_clientVersion: `MetaMask/v${version}`,
|
||||||
|
eth_sendTransaction: (payload, next, end) => {
|
||||||
|
const origin = payload.origin
|
||||||
|
const txParams = payload.params[0]
|
||||||
|
nodeify(this.txController.newUnapprovedTransaction, this.txController)(txParams, { origin }, end)
|
||||||
|
},
|
||||||
},
|
},
|
||||||
// account mgmt
|
// account mgmt
|
||||||
getAccounts: (cb) => {
|
getAccounts: (cb) => {
|
||||||
@ -256,7 +260,6 @@ module.exports = class MetamaskController extends EventEmitter {
|
|||||||
cb(null, result)
|
cb(null, result)
|
||||||
},
|
},
|
||||||
// tx signing
|
// tx signing
|
||||||
processTransaction: nodeify(async (txParams) => await this.txController.newUnapprovedTransaction(txParams), this),
|
|
||||||
// old style msg signing
|
// old style msg signing
|
||||||
processMessage: this.newUnsignedMessage.bind(this),
|
processMessage: this.newUnsignedMessage.bind(this),
|
||||||
// personal_sign msg signing
|
// personal_sign msg signing
|
||||||
|
128
development/states/tx-list-items.js
Normal file
128
development/states/tx-list-items.js
Normal file
File diff suppressed because one or more lines are too long
@ -10,87 +10,88 @@
|
|||||||
//
|
//
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
var fs = require('fs')
|
const fs = require('fs')
|
||||||
var path = require('path')
|
const path = require('path')
|
||||||
|
const localeIndex = require('../app/_locales/index.json')
|
||||||
|
|
||||||
console.log('Locale Verification')
|
console.log('Locale Verification')
|
||||||
|
|
||||||
var locale = process.argv[2]
|
const specifiedLocale = process.argv[2]
|
||||||
if (!locale || locale == '') {
|
if (specifiedLocale) {
|
||||||
console.log('Must enter a locale as argument. exitting')
|
console.log(`Verifying selected locale "${specifiedLocale}":\n\n`)
|
||||||
process.exit(1)
|
const locale = localeIndex.find(localeMeta => localeMeta.code === specifiedLocale)
|
||||||
}
|
verifyLocale({ localeMeta })
|
||||||
|
|
||||||
console.log("verifying for locale " + locale)
|
|
||||||
|
|
||||||
localeFilePath = path.join(process.cwd(), 'app', '_locales', locale, 'messages.json')
|
|
||||||
try {
|
|
||||||
localeObj = JSON.parse(fs.readFileSync(localeFilePath, 'utf8'));
|
|
||||||
} catch (e) {
|
|
||||||
if(e.code == 'ENOENT') {
|
|
||||||
console.log('Locale file not found')
|
|
||||||
} else {
|
|
||||||
console.log('Error opening your locale file: ', e)
|
|
||||||
}
|
|
||||||
process.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
englishFilePath = path.join(process.cwd(), 'app', '_locales', 'en', 'messages.json')
|
|
||||||
try {
|
|
||||||
englishObj = JSON.parse(fs.readFileSync(englishFilePath, 'utf8'));
|
|
||||||
} catch (e) {
|
|
||||||
if(e.code == 'ENOENT') {
|
|
||||||
console.log("English File not found")
|
|
||||||
} else {
|
|
||||||
console.log("Error opening english locale file: ", e)
|
|
||||||
}
|
|
||||||
process.exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
console.log('\tverifying whether all your locale strings are contained in the english one')
|
|
||||||
|
|
||||||
var counter = 0
|
|
||||||
var foundErrorA = false
|
|
||||||
var notFound = [];
|
|
||||||
Object.keys(localeObj).forEach(function(key){
|
|
||||||
if (!englishObj[key]) {
|
|
||||||
foundErrorA = true
|
|
||||||
notFound.push(key)
|
|
||||||
}
|
|
||||||
counter++
|
|
||||||
})
|
|
||||||
|
|
||||||
if (foundErrorA) {
|
|
||||||
console.log('\nThe following string(s) is(are) not found in the english locale:')
|
|
||||||
notFound.forEach(function(key) {
|
|
||||||
console.log(key)
|
|
||||||
})
|
|
||||||
} else {
|
} else {
|
||||||
console.log('\tall ' + counter +' strings declared in your locale were found in the english one')
|
console.log('Verifying all locales:\n\n')
|
||||||
}
|
localeIndex.forEach(localeMeta => {
|
||||||
|
verifyLocale({ localeMeta })
|
||||||
console.log('\n\tverifying whether your locale contains all english strings')
|
console.log('\n')
|
||||||
|
|
||||||
var counter = 0
|
|
||||||
var foundErrorB = false
|
|
||||||
var notFound = [];
|
|
||||||
Object.keys(englishObj).forEach(function(key){
|
|
||||||
if (!localeObj[key]) {
|
|
||||||
foundErrorB = true
|
|
||||||
notFound.push(key)
|
|
||||||
}
|
|
||||||
counter++
|
|
||||||
})
|
|
||||||
|
|
||||||
if (foundErrorB) {
|
|
||||||
console.log('\nThe following string(s) is(are) not found in the your locale:')
|
|
||||||
notFound.forEach(function(key) {
|
|
||||||
console.log(key)
|
|
||||||
})
|
})
|
||||||
} else {
|
|
||||||
console.log('\tall ' + counter +' english strings were found in your locale!')
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!foundErrorA && !foundErrorB) {
|
|
||||||
console.log('You are good to go')
|
|
||||||
}
|
function verifyLocale({ localeMeta }) {
|
||||||
|
const localeCode = localeMeta.code
|
||||||
|
const localeName = localeMeta.name
|
||||||
|
|
||||||
|
try {
|
||||||
|
const localeFilePath = path.join(process.cwd(), 'app', '_locales', localeCode, 'messages.json')
|
||||||
|
targetLocale = JSON.parse(fs.readFileSync(localeFilePath, 'utf8'));
|
||||||
|
} catch (e) {
|
||||||
|
if (e.code == 'ENOENT') {
|
||||||
|
console.log('Locale file not found')
|
||||||
|
} else {
|
||||||
|
console.log(`Error opening your locale ("${localeCode}") file: `, e)
|
||||||
|
}
|
||||||
|
process.exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const englishFilePath = path.join(process.cwd(), 'app', '_locales', 'en', 'messages.json')
|
||||||
|
englishLocale = JSON.parse(fs.readFileSync(englishFilePath, 'utf8'));
|
||||||
|
} catch (e) {
|
||||||
|
if(e.code == 'ENOENT') {
|
||||||
|
console.log('English File not found')
|
||||||
|
} else {
|
||||||
|
console.log('Error opening english locale file: ', e)
|
||||||
|
}
|
||||||
|
process.exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
// console.log(' verifying whether all your locale ("${localeCode}") strings are contained in the english one')
|
||||||
|
const extraItems = compareLocalesForMissingItems({ base: targetLocale, subject: englishLocale })
|
||||||
|
// console.log('\n verifying whether your locale ("${localeCode}") contains all english strings')
|
||||||
|
const missingItems = compareLocalesForMissingItems({ base: englishLocale, subject: targetLocale })
|
||||||
|
|
||||||
|
const englishEntryCount = Object.keys(englishLocale).length
|
||||||
|
const coveragePercent = 100 * (englishEntryCount - missingItems.length) / englishEntryCount
|
||||||
|
|
||||||
|
console.log(`Status of **${localeName} (${localeCode})** ${coveragePercent.toFixed(2)}% coverage:`)
|
||||||
|
|
||||||
|
if (extraItems.length) {
|
||||||
|
console.log('\nMissing from english locale:')
|
||||||
|
extraItems.forEach(function(key) {
|
||||||
|
console.log(` - [ ] ${key}`)
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
// console.log(` all ${counter} strings declared in your locale ("${localeCode}") were found in the english one`)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (missingItems.length) {
|
||||||
|
console.log(`\nMissing:`)
|
||||||
|
missingItems.forEach(function(key) {
|
||||||
|
console.log(` - [ ] ${key}`)
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
// console.log(` all ${counter} english strings were found in your locale ("${localeCode}")!`)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!extraItems.length && !missingItems.length) {
|
||||||
|
console.log('Full coverage : )')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function compareLocalesForMissingItems({ base, subject }) {
|
||||||
|
return Object.keys(base).filter((key) => !subject[key])
|
||||||
|
}
|
||||||
|
@ -6,9 +6,12 @@ The MetaMask browser extension supports new translations added in the form of ne
|
|||||||
|
|
||||||
## Adding a new Language
|
## Adding a new Language
|
||||||
|
|
||||||
Each supported language is represented by a folder in `app/_locales` whose name is that language's subtag ([look up a language subtag using this tool](https://r12a.github.io/app-subtags/)).
|
- Each supported language is represented by a folder in `app/_locales` whose name is that language's subtag (example: `app/_locales/es/`). (look up a language subtag using the [r12a "Find" tool](https://r12a.github.io/app-subtags/) or this [wikipedia list](https://en.wikipedia.org/wiki/List_of_ISO_639-1_codes)).
|
||||||
|
- Inside that folder there should be a `messages.json`.
|
||||||
|
- An easy way to start your translation is to first **make a copy** of `app/_locales/en/messages.json` (the english translation), and then **translate the `message` key** for each in-app message.
|
||||||
|
- **The `description` key** is just to add context for what the translation is about, it **does not need to be translated**.
|
||||||
|
- Add the language to the [locales index](https://github.com/MetaMask/metamask-extension/blob/master/app/_locales/index.json) `app/_locales/index.json`
|
||||||
|
|
||||||
Inside that folder there should be a `messages.json` file that follows the specified format. An easy way to start your translation is to first duplicate `app/_locales/en/messages.json` (the english translation), and then update the `message` key for each in-app message.
|
|
||||||
|
|
||||||
That's it! When MetaMask is loaded on a computer with that language set as the system language, they will see your translation instead of the default one.
|
That's it! When MetaMask is loaded on a computer with that language set as the system language, they will see your translation instead of the default one.
|
||||||
|
|
||||||
@ -20,7 +23,7 @@ To automatically see if you are missing any phrases to translate, we have a scri
|
|||||||
node development/verify-locale-strings.js $YOUR_LOCALE
|
node development/verify-locale-strings.js $YOUR_LOCALE
|
||||||
```
|
```
|
||||||
|
|
||||||
Where `$YOUR_LOCALE` is your [locale string](https://r12a.github.io/app-subtags/), i.e. the name of your language folder.
|
Where `$YOUR_LOCALE` is your locale string (example: `es`), i.e. the name of your language folder.
|
||||||
|
|
||||||
To verify that your translation works in the app, you will need to [build a local copy](https://github.com/MetaMask/metamask-extension#building-locally) of MetaMask. You will need to change your browser language, your operating system language, and restart your browser (sorry it's so much work!).
|
To verify that your translation works in the app, you will need to [build a local copy](https://github.com/MetaMask/metamask-extension#building-locally) of MetaMask. You will need to change your browser language, your operating system language, and restart your browser (sorry it's so much work!).
|
||||||
|
|
||||||
|
11
gulpfile.js
11
gulpfile.js
@ -16,7 +16,6 @@ const eslint = require('gulp-eslint')
|
|||||||
const fs = require('fs')
|
const fs = require('fs')
|
||||||
const path = require('path')
|
const path = require('path')
|
||||||
const manifest = require('./app/manifest.json')
|
const manifest = require('./app/manifest.json')
|
||||||
const gulpif = require('gulp-if')
|
|
||||||
const replace = require('gulp-replace')
|
const replace = require('gulp-replace')
|
||||||
const mkdirp = require('mkdirp')
|
const mkdirp = require('mkdirp')
|
||||||
const asyncEach = require('async/each')
|
const asyncEach = require('async/each')
|
||||||
@ -31,8 +30,6 @@ const debug = require('gulp-debug')
|
|||||||
const pify = require('pify')
|
const pify = require('pify')
|
||||||
const endOfStream = pify(require('end-of-stream'))
|
const endOfStream = pify(require('end-of-stream'))
|
||||||
|
|
||||||
const disableDebugTools = gutil.env.disableDebugTools
|
|
||||||
const debugMode = gutil.env.debug
|
|
||||||
|
|
||||||
const browserPlatforms = [
|
const browserPlatforms = [
|
||||||
'firefox',
|
'firefox',
|
||||||
@ -181,12 +178,12 @@ gulp.task('manifest:production', function() {
|
|||||||
],{base: './dist/'})
|
],{base: './dist/'})
|
||||||
|
|
||||||
// Exclude chromereload script in production:
|
// Exclude chromereload script in production:
|
||||||
.pipe(gulpif(!debugMode,jsoneditor(function(json) {
|
.pipe(jsoneditor(function(json) {
|
||||||
json.background.scripts = json.background.scripts.filter((script) => {
|
json.background.scripts = json.background.scripts.filter((script) => {
|
||||||
return !script.includes('chromereload')
|
return !script.includes('chromereload')
|
||||||
})
|
})
|
||||||
return json
|
return json
|
||||||
})))
|
}))
|
||||||
|
|
||||||
.pipe(gulp.dest('./dist/', { overwrite: true }))
|
.pipe(gulp.dest('./dist/', { overwrite: true }))
|
||||||
})
|
})
|
||||||
@ -311,6 +308,7 @@ function createTasksForBuildJsExtension({ buildJsFiles, taskPrefix, devMode, bun
|
|||||||
minifyBuild: !devMode,
|
minifyBuild: !devMode,
|
||||||
buildWithFullPaths: devMode,
|
buildWithFullPaths: devMode,
|
||||||
watch: devMode,
|
watch: devMode,
|
||||||
|
devMode,
|
||||||
}, bundleTaskOpts)
|
}, bundleTaskOpts)
|
||||||
createTasksForBuildJs({ rootDir, taskPrefix, bundleTaskOpts, destinations, buildPhase1, buildPhase2 })
|
createTasksForBuildJs({ rootDir, taskPrefix, bundleTaskOpts, destinations, buildPhase1, buildPhase2 })
|
||||||
}
|
}
|
||||||
@ -326,6 +324,7 @@ function createTasksForBuildJsMascara({ taskPrefix, devMode, bundleTaskOpts = {}
|
|||||||
minifyBuild: !devMode,
|
minifyBuild: !devMode,
|
||||||
buildWithFullPaths: devMode,
|
buildWithFullPaths: devMode,
|
||||||
watch: devMode,
|
watch: devMode,
|
||||||
|
devMode,
|
||||||
}, bundleTaskOpts)
|
}, bundleTaskOpts)
|
||||||
createTasksForBuildJs({ rootDir, taskPrefix, bundleTaskOpts, destinations, buildPhase1 })
|
createTasksForBuildJs({ rootDir, taskPrefix, bundleTaskOpts, destinations, buildPhase1 })
|
||||||
}
|
}
|
||||||
@ -541,7 +540,7 @@ function bundleTask(opts) {
|
|||||||
// convert bundle stream to gulp vinyl stream
|
// convert bundle stream to gulp vinyl stream
|
||||||
.pipe(source(opts.filename))
|
.pipe(source(opts.filename))
|
||||||
// inject variables into bundle
|
// inject variables into bundle
|
||||||
.pipe(replace('\'GULP_METAMASK_DEBUG\'', debugMode))
|
.pipe(replace('\'GULP_METAMASK_DEBUG\'', opts.devMode))
|
||||||
// buffer file contents (?)
|
// buffer file contents (?)
|
||||||
.pipe(buffer())
|
.pipe(buffer())
|
||||||
|
|
||||||
|
@ -31,6 +31,11 @@ function TransactionListItem () {
|
|||||||
TransactionListItem.prototype.showRetryButton = function () {
|
TransactionListItem.prototype.showRetryButton = function () {
|
||||||
const { transaction = {}, transactions } = this.props
|
const { transaction = {}, transactions } = this.props
|
||||||
const { status, submittedTime, txParams } = transaction
|
const { status, submittedTime, txParams } = transaction
|
||||||
|
|
||||||
|
if (!txParams) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
const currentNonce = txParams.nonce
|
const currentNonce = txParams.nonce
|
||||||
const currentNonceTxs = transactions.filter(tx => tx.txParams.nonce === currentNonce)
|
const currentNonceTxs = transactions.filter(tx => tx.txParams.nonce === currentNonce)
|
||||||
const currentNonceSubmittedTxs = currentNonceTxs.filter(tx => tx.status === 'submitted')
|
const currentNonceSubmittedTxs = currentNonceTxs.filter(tx => tx.status === 'submitted')
|
||||||
|
733
package-lock.json
generated
733
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -188,7 +188,7 @@
|
|||||||
"valid-url": "^1.0.9",
|
"valid-url": "^1.0.9",
|
||||||
"vreme": "^3.0.2",
|
"vreme": "^3.0.2",
|
||||||
"web3": "^0.20.1",
|
"web3": "^0.20.1",
|
||||||
"web3-provider-engine": "^13.5.6",
|
"web3-provider-engine": "^13.8.0",
|
||||||
"web3-stream-provider": "^3.0.1",
|
"web3-stream-provider": "^3.0.1",
|
||||||
"xtend": "^4.0.1"
|
"xtend": "^4.0.1"
|
||||||
},
|
},
|
||||||
@ -226,7 +226,6 @@
|
|||||||
"gulp": "github:gulpjs/gulp#6d71a658c61edb3090221579d8f97dbe086ba2ed",
|
"gulp": "github:gulpjs/gulp#6d71a658c61edb3090221579d8f97dbe086ba2ed",
|
||||||
"gulp-babel": "^7.0.0",
|
"gulp-babel": "^7.0.0",
|
||||||
"gulp-eslint": "^4.0.0",
|
"gulp-eslint": "^4.0.0",
|
||||||
"gulp-if": "^2.0.2",
|
|
||||||
"gulp-json-editor": "^2.2.1",
|
"gulp-json-editor": "^2.2.1",
|
||||||
"gulp-livereload": "^3.8.1",
|
"gulp-livereload": "^3.8.1",
|
||||||
"gulp-replace": "^0.6.1",
|
"gulp-replace": "^0.6.1",
|
||||||
|
61
test/integration/lib/tx-list-items.js
Normal file
61
test/integration/lib/tx-list-items.js
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
const reactTriggerChange = require('../../lib/react-trigger-change')
|
||||||
|
const {
|
||||||
|
timeout,
|
||||||
|
queryAsync,
|
||||||
|
findAsync,
|
||||||
|
} = require('../../lib/util')
|
||||||
|
|
||||||
|
QUnit.module('tx list items')
|
||||||
|
|
||||||
|
QUnit.test('renders list items successfully', (assert) => {
|
||||||
|
const done = assert.async()
|
||||||
|
runTxListItemsTest(assert).then(done).catch((err) => {
|
||||||
|
assert.notOk(err, `Error was thrown: ${err.stack}`)
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
async function runTxListItemsTest(assert, done) {
|
||||||
|
console.log('*** start runTxListItemsTest')
|
||||||
|
const selectState = await queryAsync($, 'select')
|
||||||
|
selectState.val('tx list items')
|
||||||
|
reactTriggerChange(selectState[0])
|
||||||
|
|
||||||
|
const metamaskLogo = await queryAsync($, '.left-menu-wrapper')
|
||||||
|
assert.ok(metamaskLogo[0], 'metamask logo present')
|
||||||
|
metamaskLogo[0].click()
|
||||||
|
|
||||||
|
const txListItems = await queryAsync($, '.tx-list-item')
|
||||||
|
assert.equal(txListItems.length, 8, 'all tx list items are rendered')
|
||||||
|
|
||||||
|
const unapprovedTx = txListItems[0]
|
||||||
|
assert.equal($(unapprovedTx).hasClass('tx-list-pending-item-container'), true, 'unapprovedTx has the correct class')
|
||||||
|
|
||||||
|
const retryTx = txListItems[1]
|
||||||
|
const retryTxLink = await findAsync($(retryTx), '.tx-list-item-retry-link')
|
||||||
|
assert.equal(retryTxLink[0].textContent, 'Increase the gas price on your transaction', 'retryTx has expected link')
|
||||||
|
|
||||||
|
const approvedTx = txListItems[2]
|
||||||
|
const approvedTxRenderedStatus = await findAsync($(approvedTx), '.tx-list-status')
|
||||||
|
assert.equal(approvedTxRenderedStatus[0].textContent, 'Approved', 'approvedTx has correct label')
|
||||||
|
|
||||||
|
const unapprovedMsg = txListItems[3]
|
||||||
|
const unapprovedMsgDescription = await findAsync($(unapprovedMsg), '.tx-list-account')
|
||||||
|
assert.equal(unapprovedMsgDescription[0].textContent, 'Signature Request', 'unapprovedMsg has correct description')
|
||||||
|
|
||||||
|
const failedTx = txListItems[4]
|
||||||
|
const failedTxRenderedStatus = await findAsync($(failedTx), '.tx-list-status')
|
||||||
|
assert.equal(failedTxRenderedStatus[0].textContent, 'Failed', 'failedTx has correct label')
|
||||||
|
|
||||||
|
const shapeShiftTx = txListItems[5]
|
||||||
|
const shapeShiftTxStatus = await findAsync($(shapeShiftTx), '.flex-column div:eq(1)')
|
||||||
|
assert.equal(shapeShiftTxStatus[0].textContent, 'No deposits received', 'shapeShiftTx has correct status')
|
||||||
|
|
||||||
|
const confirmedTokenTx = txListItems[6]
|
||||||
|
const confirmedTokenTxAddress = await findAsync($(confirmedTokenTx), '.tx-list-account')
|
||||||
|
assert.equal(confirmedTokenTxAddress[0].textContent, '0xe7884118...81a9', 'confirmedTokenTx has correct address')
|
||||||
|
|
||||||
|
const rejectedTx = txListItems[7]
|
||||||
|
const rejectedTxRenderedStatus = await findAsync($(rejectedTx), '.tx-list-status')
|
||||||
|
assert.equal(rejectedTxRenderedStatus[0].textContent, 'Rejected', 'rejectedTx has correct label')
|
||||||
|
}
|
@ -65,6 +65,7 @@ function mapDispatchToProps (dispatch) {
|
|||||||
updateGasLimit: newGasLimit => dispatch(actions.updateGasLimit(newGasLimit)),
|
updateGasLimit: newGasLimit => dispatch(actions.updateGasLimit(newGasLimit)),
|
||||||
updateGasTotal: newGasTotal => dispatch(actions.updateGasTotal(newGasTotal)),
|
updateGasTotal: newGasTotal => dispatch(actions.updateGasTotal(newGasTotal)),
|
||||||
updateSendAmount: newAmount => dispatch(actions.updateSendAmount(newAmount)),
|
updateSendAmount: newAmount => dispatch(actions.updateSendAmount(newAmount)),
|
||||||
|
updateSendErrors: error => dispatch(actions.updateSendErrors(error)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -112,6 +113,7 @@ CustomizeGasModal.prototype.save = function (gasPrice, gasLimit, gasTotal) {
|
|||||||
selectedToken,
|
selectedToken,
|
||||||
balance,
|
balance,
|
||||||
updateSendAmount,
|
updateSendAmount,
|
||||||
|
updateSendErrors,
|
||||||
} = this.props
|
} = this.props
|
||||||
|
|
||||||
if (maxModeOn && !selectedToken) {
|
if (maxModeOn && !selectedToken) {
|
||||||
@ -126,6 +128,7 @@ CustomizeGasModal.prototype.save = function (gasPrice, gasLimit, gasTotal) {
|
|||||||
updateGasPrice(ethUtil.addHexPrefix(gasPrice))
|
updateGasPrice(ethUtil.addHexPrefix(gasPrice))
|
||||||
updateGasLimit(ethUtil.addHexPrefix(gasLimit))
|
updateGasLimit(ethUtil.addHexPrefix(gasLimit))
|
||||||
updateGasTotal(ethUtil.addHexPrefix(gasTotal))
|
updateGasTotal(ethUtil.addHexPrefix(gasTotal))
|
||||||
|
updateSendErrors({ insufficientFunds: false })
|
||||||
hideModal()
|
hideModal()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -105,9 +105,8 @@ IdenticonComponent.prototype.componentDidUpdate = function () {
|
|||||||
function _generateBlockie (container, address, diameter) {
|
function _generateBlockie (container, address, diameter) {
|
||||||
const img = new Image()
|
const img = new Image()
|
||||||
img.src = toDataUrl(address)
|
img.src = toDataUrl(address)
|
||||||
const dia = !diameter || diameter < 50 ? 50 : diameter
|
img.height = diameter
|
||||||
img.height = dia * 1.25
|
img.width = diameter
|
||||||
img.width = dia * 1.25
|
|
||||||
container.appendChild(img)
|
container.appendChild(img)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8,11 +8,16 @@ const clone = require('clone')
|
|||||||
const ethUtil = require('ethereumjs-util')
|
const ethUtil = require('ethereumjs-util')
|
||||||
const BN = ethUtil.BN
|
const BN = ethUtil.BN
|
||||||
const hexToBn = require('../../../../app/scripts/lib/hex-to-bn')
|
const hexToBn = require('../../../../app/scripts/lib/hex-to-bn')
|
||||||
|
const classnames = require('classnames')
|
||||||
const {
|
const {
|
||||||
conversionUtil,
|
conversionUtil,
|
||||||
addCurrencies,
|
addCurrencies,
|
||||||
multiplyCurrencies,
|
multiplyCurrencies,
|
||||||
} = require('../../conversion-util')
|
} = require('../../conversion-util')
|
||||||
|
const {
|
||||||
|
getGasTotal,
|
||||||
|
isBalanceSufficient,
|
||||||
|
} = require('../send/send-utils')
|
||||||
const GasFeeDisplay = require('../send/gas-fee-display-v2')
|
const GasFeeDisplay = require('../send/gas-fee-display-v2')
|
||||||
const SenderToRecipient = require('../sender-to-recipient')
|
const SenderToRecipient = require('../sender-to-recipient')
|
||||||
const NetworkDisplay = require('../network-display')
|
const NetworkDisplay = require('../network-display')
|
||||||
@ -35,12 +40,14 @@ function mapStateToProps (state) {
|
|||||||
} = state.metamask
|
} = state.metamask
|
||||||
const accounts = state.metamask.accounts
|
const accounts = state.metamask.accounts
|
||||||
const selectedAddress = state.metamask.selectedAddress || Object.keys(accounts)[0]
|
const selectedAddress = state.metamask.selectedAddress || Object.keys(accounts)[0]
|
||||||
|
const { balance } = accounts[selectedAddress]
|
||||||
return {
|
return {
|
||||||
conversionRate,
|
conversionRate,
|
||||||
identities,
|
identities,
|
||||||
selectedAddress,
|
selectedAddress,
|
||||||
currentCurrency,
|
currentCurrency,
|
||||||
send,
|
send,
|
||||||
|
balance,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -91,6 +98,7 @@ function mapDispatchToProps (dispatch) {
|
|||||||
}))
|
}))
|
||||||
dispatch(actions.showModal({ name: 'CUSTOMIZE_GAS' }))
|
dispatch(actions.showModal({ name: 'CUSTOMIZE_GAS' }))
|
||||||
},
|
},
|
||||||
|
updateSendErrors: error => dispatch(actions.updateSendErrors(error)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -101,6 +109,18 @@ function ConfirmSendEther () {
|
|||||||
this.onSubmit = this.onSubmit.bind(this)
|
this.onSubmit = this.onSubmit.bind(this)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ConfirmSendEther.prototype.componentWillMount = function () {
|
||||||
|
const { updateSendErrors } = this.props
|
||||||
|
const txMeta = this.gatherTxMeta()
|
||||||
|
const balanceIsSufficient = this.isBalanceSufficient(txMeta)
|
||||||
|
|
||||||
|
updateSendErrors({
|
||||||
|
insufficientFunds: balanceIsSufficient
|
||||||
|
? false
|
||||||
|
: this.context.t('insufficientFunds'),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
ConfirmSendEther.prototype.getAmount = function () {
|
ConfirmSendEther.prototype.getAmount = function () {
|
||||||
const { conversionRate, currentCurrency } = this.props
|
const { conversionRate, currentCurrency } = this.props
|
||||||
const txMeta = this.gatherTxMeta()
|
const txMeta = this.gatherTxMeta()
|
||||||
@ -223,7 +243,12 @@ ConfirmSendEther.prototype.render = function () {
|
|||||||
conversionRate,
|
conversionRate,
|
||||||
currentCurrency: convertedCurrency,
|
currentCurrency: convertedCurrency,
|
||||||
showCustomizeGasModal,
|
showCustomizeGasModal,
|
||||||
send: { gasTotal, gasLimit: sendGasLimit, gasPrice: sendGasPrice },
|
send: {
|
||||||
|
gasTotal,
|
||||||
|
gasLimit: sendGasLimit,
|
||||||
|
gasPrice: sendGasPrice,
|
||||||
|
errors,
|
||||||
|
},
|
||||||
} = this.props
|
} = this.props
|
||||||
const txMeta = this.gatherTxMeta()
|
const txMeta = this.gatherTxMeta()
|
||||||
const txParams = txMeta.txParams || {}
|
const txParams = txMeta.txParams || {}
|
||||||
@ -331,7 +356,12 @@ ConfirmSendEther.prototype.render = function () {
|
|||||||
]),
|
]),
|
||||||
|
|
||||||
h('section.flex-row.flex-center.confirm-screen-row.confirm-screen-total-box ', [
|
h('section.flex-row.flex-center.confirm-screen-row.confirm-screen-total-box ', [
|
||||||
h('div.confirm-screen-section-column', [
|
h('div', {
|
||||||
|
className: classnames({
|
||||||
|
'confirm-screen-section-column--with-error': errors['insufficientFunds'],
|
||||||
|
'confirm-screen-section-column': !errors['insufficientFunds'],
|
||||||
|
}),
|
||||||
|
}, [
|
||||||
h('span.confirm-screen-label', [ this.context.t('total') + ' ' ]),
|
h('span.confirm-screen-label', [ this.context.t('total') + ' ' ]),
|
||||||
h('div.confirm-screen-total-box__subtitle', [ this.context.t('amountPlusGas') ]),
|
h('div.confirm-screen-total-box__subtitle', [ this.context.t('amountPlusGas') ]),
|
||||||
]),
|
]),
|
||||||
@ -340,6 +370,8 @@ ConfirmSendEther.prototype.render = function () {
|
|||||||
h('div.confirm-screen-row-info', `${totalInFIAT} ${currentCurrency.toUpperCase()}`),
|
h('div.confirm-screen-row-info', `${totalInFIAT} ${currentCurrency.toUpperCase()}`),
|
||||||
h('div.confirm-screen-row-detail', `${totalInETH} ETH`),
|
h('div.confirm-screen-row-detail', `${totalInETH} ETH`),
|
||||||
]),
|
]),
|
||||||
|
|
||||||
|
this.renderErrorMessage('insufficientFunds'),
|
||||||
]),
|
]),
|
||||||
]),
|
]),
|
||||||
|
|
||||||
@ -444,16 +476,28 @@ ConfirmSendEther.prototype.render = function () {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ConfirmSendEther.prototype.renderErrorMessage = function (message) {
|
||||||
|
const { send: { errors } } = this.props
|
||||||
|
|
||||||
|
return errors[message]
|
||||||
|
? h('div.confirm-screen-error', [ errors[message] ])
|
||||||
|
: null
|
||||||
|
}
|
||||||
|
|
||||||
ConfirmSendEther.prototype.onSubmit = function (event) {
|
ConfirmSendEther.prototype.onSubmit = function (event) {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
|
const { updateSendErrors } = this.props
|
||||||
const txMeta = this.gatherTxMeta()
|
const txMeta = this.gatherTxMeta()
|
||||||
const valid = this.checkValidity()
|
const valid = this.checkValidity()
|
||||||
|
const balanceIsSufficient = this.isBalanceSufficient(txMeta)
|
||||||
this.setState({ valid, submitting: true })
|
this.setState({ valid, submitting: true })
|
||||||
|
|
||||||
if (valid && this.verifyGasParams()) {
|
if (valid && this.verifyGasParams() && balanceIsSufficient) {
|
||||||
this.props.sendTransaction(txMeta, event)
|
this.props.sendTransaction(txMeta, event)
|
||||||
|
} else if (!balanceIsSufficient) {
|
||||||
|
updateSendErrors({ insufficientFunds: this.context.t('insufficientFunds') })
|
||||||
} else {
|
} else {
|
||||||
this.props.dispatch(actions.displayWarning(this.context.t('invalidGasParams')))
|
updateSendErrors({ invalidGasParams: this.context.t('invalidGasParams') })
|
||||||
this.setState({ submitting: false })
|
this.setState({ submitting: false })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -465,6 +509,28 @@ ConfirmSendEther.prototype.cancel = function (event, txMeta) {
|
|||||||
cancelTransaction(txMeta)
|
cancelTransaction(txMeta)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ConfirmSendEther.prototype.isBalanceSufficient = function (txMeta) {
|
||||||
|
const {
|
||||||
|
balance,
|
||||||
|
conversionRate,
|
||||||
|
} = this.props
|
||||||
|
const {
|
||||||
|
txParams: {
|
||||||
|
gas,
|
||||||
|
gasPrice,
|
||||||
|
value: amount,
|
||||||
|
},
|
||||||
|
} = txMeta
|
||||||
|
const gasTotal = getGasTotal(gas, gasPrice)
|
||||||
|
|
||||||
|
return isBalanceSufficient({
|
||||||
|
amount,
|
||||||
|
gasTotal,
|
||||||
|
balance,
|
||||||
|
conversionRate,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
ConfirmSendEther.prototype.checkValidity = function () {
|
ConfirmSendEther.prototype.checkValidity = function () {
|
||||||
const form = this.getFormEl()
|
const form = this.getFormEl()
|
||||||
const valid = form.checkValidity()
|
const valid = form.checkValidity()
|
||||||
|
@ -17,9 +17,14 @@ const {
|
|||||||
multiplyCurrencies,
|
multiplyCurrencies,
|
||||||
addCurrencies,
|
addCurrencies,
|
||||||
} = require('../../conversion-util')
|
} = require('../../conversion-util')
|
||||||
|
const {
|
||||||
|
getGasTotal,
|
||||||
|
isBalanceSufficient,
|
||||||
|
} = require('../send/send-utils')
|
||||||
const {
|
const {
|
||||||
calcTokenAmount,
|
calcTokenAmount,
|
||||||
} = require('../../token-util')
|
} = require('../../token-util')
|
||||||
|
const classnames = require('classnames')
|
||||||
|
|
||||||
const { MIN_GAS_PRICE_HEX } = require('../send/send-constants')
|
const { MIN_GAS_PRICE_HEX } = require('../send/send-constants')
|
||||||
|
|
||||||
@ -46,9 +51,10 @@ function mapStateToProps (state, ownProps) {
|
|||||||
identities,
|
identities,
|
||||||
currentCurrency,
|
currentCurrency,
|
||||||
} = state.metamask
|
} = state.metamask
|
||||||
|
const accounts = state.metamask.accounts
|
||||||
const selectedAddress = getSelectedAddress(state)
|
const selectedAddress = getSelectedAddress(state)
|
||||||
const tokenExchangeRate = getTokenExchangeRate(state, symbol)
|
const tokenExchangeRate = getTokenExchangeRate(state, symbol)
|
||||||
|
const { balance } = accounts[selectedAddress]
|
||||||
return {
|
return {
|
||||||
conversionRate,
|
conversionRate,
|
||||||
identities,
|
identities,
|
||||||
@ -58,6 +64,7 @@ function mapStateToProps (state, ownProps) {
|
|||||||
currentCurrency: currentCurrency.toUpperCase(),
|
currentCurrency: currentCurrency.toUpperCase(),
|
||||||
send: state.metamask.send,
|
send: state.metamask.send,
|
||||||
tokenContract: getSelectedTokenContract(state),
|
tokenContract: getSelectedTokenContract(state),
|
||||||
|
balance,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -129,6 +136,7 @@ function mapDispatchToProps (dispatch, ownProps) {
|
|||||||
}))
|
}))
|
||||||
dispatch(actions.showModal({ name: 'CUSTOMIZE_GAS' }))
|
dispatch(actions.showModal({ name: 'CUSTOMIZE_GAS' }))
|
||||||
},
|
},
|
||||||
|
updateSendErrors: error => dispatch(actions.updateSendErrors(error)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -140,12 +148,20 @@ function ConfirmSendToken () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ConfirmSendToken.prototype.componentWillMount = function () {
|
ConfirmSendToken.prototype.componentWillMount = function () {
|
||||||
const { tokenContract, selectedAddress } = this.props
|
const { tokenContract, selectedAddress, updateSendErrors} = this.props
|
||||||
|
const txMeta = this.gatherTxMeta()
|
||||||
|
const balanceIsSufficient = this.isBalanceSufficient(txMeta)
|
||||||
tokenContract && tokenContract
|
tokenContract && tokenContract
|
||||||
.balanceOf(selectedAddress)
|
.balanceOf(selectedAddress)
|
||||||
.then(usersToken => {
|
.then(usersToken => {
|
||||||
})
|
})
|
||||||
this.props.updateTokenExchangeRate()
|
this.props.updateTokenExchangeRate()
|
||||||
|
|
||||||
|
updateSendErrors({
|
||||||
|
insufficientFunds: balanceIsSufficient
|
||||||
|
? false
|
||||||
|
: this.context.t('insufficientFunds'),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
ConfirmSendToken.prototype.getAmount = function () {
|
ConfirmSendToken.prototype.getAmount = function () {
|
||||||
@ -306,7 +322,7 @@ ConfirmSendToken.prototype.renderGasFee = function () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ConfirmSendToken.prototype.renderTotalPlusGas = function () {
|
ConfirmSendToken.prototype.renderTotalPlusGas = function () {
|
||||||
const { token: { symbol }, currentCurrency } = this.props
|
const { token: { symbol }, currentCurrency, send: { errors } } = this.props
|
||||||
const { fiat: fiatAmount, token: tokenAmount } = this.getAmount()
|
const { fiat: fiatAmount, token: tokenAmount } = this.getAmount()
|
||||||
const { fiat: fiatGas, token: tokenGas } = this.getGasFee()
|
const { fiat: fiatGas, token: tokenGas } = this.getGasFee()
|
||||||
|
|
||||||
@ -326,7 +342,12 @@ ConfirmSendToken.prototype.renderTotalPlusGas = function () {
|
|||||||
)
|
)
|
||||||
: (
|
: (
|
||||||
h('section.flex-row.flex-center.confirm-screen-row.confirm-screen-total-box ', [
|
h('section.flex-row.flex-center.confirm-screen-row.confirm-screen-total-box ', [
|
||||||
h('div.confirm-screen-section-column', [
|
h('div', {
|
||||||
|
className: classnames({
|
||||||
|
'confirm-screen-section-column--with-error': errors['insufficientFunds'],
|
||||||
|
'confirm-screen-section-column': !errors['insufficientFunds'],
|
||||||
|
}),
|
||||||
|
}, [
|
||||||
h('span.confirm-screen-label', [ this.context.t('total') + ' ' ]),
|
h('span.confirm-screen-label', [ this.context.t('total') + ' ' ]),
|
||||||
h('div.confirm-screen-total-box__subtitle', [ this.context.t('amountPlusGas') ]),
|
h('div.confirm-screen-total-box__subtitle', [ this.context.t('amountPlusGas') ]),
|
||||||
]),
|
]),
|
||||||
@ -335,10 +356,20 @@ ConfirmSendToken.prototype.renderTotalPlusGas = function () {
|
|||||||
h('div.confirm-screen-row-info', `${tokenAmount} ${symbol}`),
|
h('div.confirm-screen-row-info', `${tokenAmount} ${symbol}`),
|
||||||
h('div.confirm-screen-row-detail', `+ ${fiatGas} ${currentCurrency} ${this.context.t('gas')}`),
|
h('div.confirm-screen-row-detail', `+ ${fiatGas} ${currentCurrency} ${this.context.t('gas')}`),
|
||||||
]),
|
]),
|
||||||
|
|
||||||
|
this.renderErrorMessage('insufficientFunds'),
|
||||||
])
|
])
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ConfirmSendToken.prototype.renderErrorMessage = function (message) {
|
||||||
|
const { send: { errors } } = this.props
|
||||||
|
|
||||||
|
return errors[message]
|
||||||
|
? h('div.confirm-screen-error', [ errors[message] ])
|
||||||
|
: null
|
||||||
|
}
|
||||||
|
|
||||||
ConfirmSendToken.prototype.render = function () {
|
ConfirmSendToken.prototype.render = function () {
|
||||||
const { editTransaction } = this.props
|
const { editTransaction } = this.props
|
||||||
const txMeta = this.gatherTxMeta()
|
const txMeta = this.gatherTxMeta()
|
||||||
@ -455,18 +486,44 @@ ConfirmSendToken.prototype.render = function () {
|
|||||||
|
|
||||||
ConfirmSendToken.prototype.onSubmit = function (event) {
|
ConfirmSendToken.prototype.onSubmit = function (event) {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
|
const { updateSendErrors } = this.props
|
||||||
const txMeta = this.gatherTxMeta()
|
const txMeta = this.gatherTxMeta()
|
||||||
const valid = this.checkValidity()
|
const valid = this.checkValidity()
|
||||||
|
const balanceIsSufficient = this.isBalanceSufficient(txMeta)
|
||||||
this.setState({ valid, submitting: true })
|
this.setState({ valid, submitting: true })
|
||||||
|
|
||||||
if (valid && this.verifyGasParams()) {
|
if (valid && this.verifyGasParams() && balanceIsSufficient) {
|
||||||
this.props.sendTransaction(txMeta, event)
|
this.props.sendTransaction(txMeta, event)
|
||||||
|
} else if (!balanceIsSufficient) {
|
||||||
|
updateSendErrors({ insufficientFunds: this.context.t('insufficientFunds') })
|
||||||
} else {
|
} else {
|
||||||
this.props.dispatch(actions.displayWarning(this.context.t('invalidGasParams')))
|
updateSendErrors({ invalidGasParams: this.context.t('invalidGasParams') })
|
||||||
this.setState({ submitting: false })
|
this.setState({ submitting: false })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ConfirmSendToken.prototype.isBalanceSufficient = function (txMeta) {
|
||||||
|
const {
|
||||||
|
balance,
|
||||||
|
conversionRate,
|
||||||
|
} = this.props
|
||||||
|
const {
|
||||||
|
txParams: {
|
||||||
|
gas,
|
||||||
|
gasPrice,
|
||||||
|
},
|
||||||
|
} = txMeta
|
||||||
|
const gasTotal = getGasTotal(gas, gasPrice)
|
||||||
|
|
||||||
|
return isBalanceSufficient({
|
||||||
|
amount: '0',
|
||||||
|
gasTotal,
|
||||||
|
balance,
|
||||||
|
conversionRate,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
ConfirmSendToken.prototype.cancel = function (event, txMeta) {
|
ConfirmSendToken.prototype.cancel = function (event, txMeta) {
|
||||||
event.preventDefault()
|
event.preventDefault()
|
||||||
const { cancelTransaction } = this.props
|
const { cancelTransaction } = this.props
|
||||||
|
@ -2,6 +2,7 @@ const {
|
|||||||
addCurrencies,
|
addCurrencies,
|
||||||
conversionUtil,
|
conversionUtil,
|
||||||
conversionGTE,
|
conversionGTE,
|
||||||
|
multiplyCurrencies,
|
||||||
} = require('../../conversion-util')
|
} = require('../../conversion-util')
|
||||||
const {
|
const {
|
||||||
calcTokenAmount,
|
calcTokenAmount,
|
||||||
@ -31,7 +32,7 @@ function isBalanceSufficient ({
|
|||||||
{
|
{
|
||||||
value: totalAmount,
|
value: totalAmount,
|
||||||
fromNumericBase: 'hex',
|
fromNumericBase: 'hex',
|
||||||
conversionRate: amountConversionRate,
|
conversionRate: amountConversionRate || conversionRate,
|
||||||
fromCurrency: primaryCurrency,
|
fromCurrency: primaryCurrency,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
@ -62,7 +63,16 @@ function isTokenBalanceSufficient ({
|
|||||||
return tokenBalanceIsSufficient
|
return tokenBalanceIsSufficient
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getGasTotal (gasLimit, gasPrice) {
|
||||||
|
return multiplyCurrencies(gasLimit, gasPrice, {
|
||||||
|
toNumericBase: 'hex',
|
||||||
|
multiplicandBase: 16,
|
||||||
|
multiplierBase: 16,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
|
getGasTotal,
|
||||||
isBalanceSufficient,
|
isBalanceSufficient,
|
||||||
isTokenBalanceSufficient,
|
isTokenBalanceSufficient,
|
||||||
}
|
}
|
||||||
|
@ -68,20 +68,24 @@ TxListItem.prototype.getAddressText = function () {
|
|||||||
const {
|
const {
|
||||||
address,
|
address,
|
||||||
txParams = {},
|
txParams = {},
|
||||||
|
isMsg,
|
||||||
} = this.props
|
} = this.props
|
||||||
|
|
||||||
const decodedData = txParams.data && abiDecoder.decodeMethod(txParams.data)
|
const decodedData = txParams.data && abiDecoder.decodeMethod(txParams.data)
|
||||||
const { name: txDataName, params = [] } = decodedData || {}
|
const { name: txDataName, params = [] } = decodedData || {}
|
||||||
const { value } = params[0] || {}
|
const { value } = params[0] || {}
|
||||||
|
|
||||||
switch (txDataName) {
|
let addressText
|
||||||
case 'transfer':
|
if (txDataName === 'transfer' || address) {
|
||||||
return `${value.slice(0, 10)}...${value.slice(-4)}`
|
const addressToRender = txDataName === 'transfer' ? value : address
|
||||||
default:
|
addressText = `${addressToRender.slice(0, 10)}...${addressToRender.slice(-4)}`
|
||||||
return address
|
} else if (isMsg) {
|
||||||
? `${address.slice(0, 10)}...${address.slice(-4)}`
|
addressText = this.context.t('sigRequest')
|
||||||
: this.context.t('contractDeployment')
|
} else {
|
||||||
|
addressText = this.context.t('contractDeployment')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return addressText
|
||||||
}
|
}
|
||||||
|
|
||||||
TxListItem.prototype.getSendEtherTotal = function () {
|
TxListItem.prototype.getSendEtherTotal = function () {
|
||||||
@ -191,6 +195,9 @@ TxListItem.prototype.showRetryButton = function () {
|
|||||||
transactionId,
|
transactionId,
|
||||||
txParams,
|
txParams,
|
||||||
} = this.props
|
} = this.props
|
||||||
|
if (!txParams) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
const currentNonce = txParams.nonce
|
const currentNonce = txParams.nonce
|
||||||
const currentNonceTxs = selectedAddressTxList.filter(tx => tx.txParams.nonce === currentNonce)
|
const currentNonceTxs = selectedAddressTxList.filter(tx => tx.txParams.nonce === currentNonce)
|
||||||
const currentNonceSubmittedTxs = currentNonceTxs.filter(tx => tx.status === 'submitted')
|
const currentNonceSubmittedTxs = currentNonceTxs.filter(tx => tx.status === 'submitted')
|
||||||
|
@ -77,9 +77,9 @@ TxList.prototype.renderTransactionListItem = function (transaction, conversionRa
|
|||||||
|
|
||||||
const props = {
|
const props = {
|
||||||
dateString: formatDate(transaction.time),
|
dateString: formatDate(transaction.time),
|
||||||
address: transaction.txParams.to,
|
address: transaction.txParams && transaction.txParams.to,
|
||||||
transactionStatus: transaction.status,
|
transactionStatus: transaction.status,
|
||||||
transactionAmount: transaction.txParams.value,
|
transactionAmount: transaction.txParams && transaction.txParams.value,
|
||||||
transactionId: transaction.id,
|
transactionId: transaction.id,
|
||||||
transactionHash: transaction.hash,
|
transactionHash: transaction.hash,
|
||||||
transactionNetworkId: transaction.metamaskNetworkId,
|
transactionNetworkId: transaction.metamaskNetworkId,
|
||||||
@ -101,6 +101,7 @@ TxList.prototype.renderTransactionListItem = function (transaction, conversionRa
|
|||||||
const opts = {
|
const opts = {
|
||||||
key: transactionId || transactionHash,
|
key: transactionId || transactionHash,
|
||||||
txParams: transaction.txParams,
|
txParams: transaction.txParams,
|
||||||
|
isMsg: Boolean(transaction.msgParams),
|
||||||
transactionStatus,
|
transactionStatus,
|
||||||
transactionId,
|
transactionId,
|
||||||
dateString,
|
dateString,
|
||||||
|
@ -266,6 +266,7 @@ section .confirm-screen-account-number,
|
|||||||
|
|
||||||
.confirm-screen-total-box {
|
.confirm-screen-total-box {
|
||||||
background-color: $wild-sand;
|
background-color: $wild-sand;
|
||||||
|
position: relative;
|
||||||
|
|
||||||
.confirm-screen-label {
|
.confirm-screen-label {
|
||||||
line-height: 21px;
|
line-height: 21px;
|
||||||
@ -287,6 +288,30 @@ section .confirm-screen-account-number,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.confirm-screen-error {
|
||||||
|
font-size: 12px;
|
||||||
|
line-height: 12px;
|
||||||
|
color: #f00;
|
||||||
|
position: absolute;
|
||||||
|
right: 12px;
|
||||||
|
width: 80px;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
.confirm-screen-row.confirm-screen-total-box {
|
||||||
|
.confirm-screen-section-column--with-error {
|
||||||
|
flex: 0.6;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media screen and (max-width: 379px) {
|
||||||
|
.confirm-screen-row.confirm-screen-total-box {
|
||||||
|
.confirm-screen-section-column--with-error {
|
||||||
|
flex: 0.4;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.confirm-screen-confirm-button {
|
.confirm-screen-confirm-button {
|
||||||
height: 50px;
|
height: 50px;
|
||||||
border-radius: 4px;
|
border-radius: 4px;
|
||||||
|
@ -27,6 +27,7 @@ const {
|
|||||||
const {
|
const {
|
||||||
isBalanceSufficient,
|
isBalanceSufficient,
|
||||||
isTokenBalanceSufficient,
|
isTokenBalanceSufficient,
|
||||||
|
getGasTotal,
|
||||||
} = require('./components/send/send-utils')
|
} = require('./components/send/send-utils')
|
||||||
const { isValidAddress } = require('./util')
|
const { isValidAddress } = require('./util')
|
||||||
|
|
||||||
@ -132,7 +133,7 @@ SendTransactionScreen.prototype.updateGas = function () {
|
|||||||
estimateGas(estimateGasParams),
|
estimateGas(estimateGasParams),
|
||||||
])
|
])
|
||||||
.then(([gasPrice, gas]) => {
|
.then(([gasPrice, gas]) => {
|
||||||
const newGasTotal = this.getGasTotal(gas, gasPrice)
|
const newGasTotal = getGasTotal(gas, gasPrice)
|
||||||
updateGasTotal(newGasTotal)
|
updateGasTotal(newGasTotal)
|
||||||
this.setState({ gasLoadingError: false })
|
this.setState({ gasLoadingError: false })
|
||||||
})
|
})
|
||||||
@ -140,19 +141,11 @@ SendTransactionScreen.prototype.updateGas = function () {
|
|||||||
this.setState({ gasLoadingError: true })
|
this.setState({ gasLoadingError: true })
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
const newGasTotal = this.getGasTotal(gasLimit, gasPrice)
|
const newGasTotal = getGasTotal(gasLimit, gasPrice)
|
||||||
updateGasTotal(newGasTotal)
|
updateGasTotal(newGasTotal)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
SendTransactionScreen.prototype.getGasTotal = function (gasLimit, gasPrice) {
|
|
||||||
return multiplyCurrencies(gasLimit, gasPrice, {
|
|
||||||
toNumericBase: 'hex',
|
|
||||||
multiplicandBase: 16,
|
|
||||||
multiplierBase: 16,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
SendTransactionScreen.prototype.componentDidUpdate = function (prevProps) {
|
SendTransactionScreen.prototype.componentDidUpdate = function (prevProps) {
|
||||||
const {
|
const {
|
||||||
from: { balance },
|
from: { balance },
|
||||||
@ -642,6 +635,10 @@ SendTransactionScreen.prototype.onSubmit = function (event) {
|
|||||||
txParams.to = to
|
txParams.to = to
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Object.keys(txParams).forEach(key => {
|
||||||
|
txParams[key] = ethUtil.addHexPrefix(txParams[key])
|
||||||
|
})
|
||||||
|
|
||||||
selectedToken
|
selectedToken
|
||||||
? signTokenTx(selectedToken.address, to, amount, txParams)
|
? signTokenTx(selectedToken.address, to, amount, txParams)
|
||||||
: signTx(txParams)
|
: signTx(txParams)
|
||||||
|
Loading…
Reference in New Issue
Block a user