From 0cc521e130f0bd6010ad44659665ecf34352e58b Mon Sep 17 00:00:00 2001 From: Jacky Chan Date: Tue, 29 Aug 2017 03:40:47 -0700 Subject: Implement Buy Ether Screen UI --- mascara/src/app/first-time/buy-ether-screen.js | 184 +++++++++++++++++++++++++ mascara/src/app/first-time/index.css | 133 +++++++++++++++++- mascara/src/app/first-time/index.js | 9 +- 3 files changed, 318 insertions(+), 8 deletions(-) create mode 100644 mascara/src/app/first-time/buy-ether-screen.js diff --git a/mascara/src/app/first-time/buy-ether-screen.js b/mascara/src/app/first-time/buy-ether-screen.js new file mode 100644 index 000000000..395a11be5 --- /dev/null +++ b/mascara/src/app/first-time/buy-ether-screen.js @@ -0,0 +1,184 @@ +import React, {Component, PropTypes} from 'react' +import classnames from 'classnames' +import {connect} from 'react-redux' +import {qrcode} from 'qrcode-npm' +import Identicon from '../../../../ui/app/components/identicon' + +class BuyEtherScreen extends Component { + static OPTION_VALUES = { + COINBASE: 'coinbase', + SHAPESHIFT: 'shapeshift', + QR_CODE: 'qr_code', + }; + + static OPTIONS = [ + { + name: 'Buy with Dollars', + value: BuyEtherScreen.OPTION_VALUES.COINBASE + }, + { + name: 'Buy with Bitcoin', + value: BuyEtherScreen.OPTION_VALUES.SHAPESHIFT + }, + { + name: 'Deposit Ether', + value: BuyEtherScreen.OPTION_VALUES.QR_CODE + }, + ]; + + static propTypes = { + address: PropTypes.string, + } + + state = { + selectedOption: BuyEtherScreen.OPTION_VALUES.COINBASE, + } + + renderCoinbaseLogo() { + return ( + + + + + + + + + + + + + + + ); + } + + renderShapeShiftLogo() { + return ( +
+ ) + } + + renderContent() { + const { OPTION_VALUES } = BuyEtherScreen; + const { address } = this.props; + + switch (this.state.selectedOption) { + case OPTION_VALUES.COINBASE: + return ( +
+
{this.renderCoinbaseLogo()}
+
Coinbase is the world’s most popular way to buy and sell bitcoin, ethereum, and litecoin.
+ What is Ethereum? +
+ +
or
+ +
+
+ ) + case OPTION_VALUES.SHAPESHIFT: + return ( +
+
{this.renderShapeShiftLogo()}
+
Trade any leading blockchain asset for any other. Protection by Design. No Account Needed.
+
+ +
or
+ +
+
+ ) + case OPTION_VALUES.QR_CODE: + const qrImage = qrcode(4, 'M') + qrImage.addData(address) + qrImage.make() + return ( +
+
+
Deposit Ether directly into your account.
+
(This is the account address that MetaMask created for you to recieve funds.)
+
+ +
or
+ +
+
+ ) + default: + return null; + } + } + + render() { + const { OPTIONS } = BuyEtherScreen; + const { selectedOption } = this.state; + + return ( +
+ +
Buy Ether
+
+ MetaMask works best if you have Ether in your account to pay for transaction gas fees and more. To get Ether, choose from one of these methods. +
+
+
Purchasing Options
+
+
+ {OPTIONS.map(({ name, value }) => ( +
this.setState({ selectedOption: value })} + > +
{name}
+ {value === selectedOption && ( + + + + )} +
+ ))} +
+
+ {this.renderContent()} +
+
+
+
+ ) + } +} + +export default connect( + ({ metamask: { identities } }) => ({ + address: Object.entries(identities) + .map(([key]) => key)[0] + }) +)(BuyEtherScreen) diff --git a/mascara/src/app/first-time/index.css b/mascara/src/app/first-time/index.css index da8f801e8..8e836e1da 100644 --- a/mascara/src/app/first-time/index.css +++ b/mascara/src/app/first-time/index.css @@ -1,5 +1,3 @@ -$primary - .first-time-flow { height: 100vh; width: 100vw; @@ -10,7 +8,8 @@ $primary .unique-image, .tou, .backup-phrase, -.import-account { +.import-account, +.buy-ether { display: flex; flex-flow: column nowrap; margin: 67px 0 0 146px; @@ -29,7 +28,8 @@ $primary .unique-image__title, .tou__title, .backup-phrase__title, -.import-account__title { +.import-account__title, +.buy-ether__title { width: 280px; color: #1B344D; font-size: 40px; @@ -53,18 +53,28 @@ $primary .unique-image__title, .tou__title, -.backup-phrase__title { +.backup-phrase__title, +.buy-ether__title { margin-top: 24px; } .unique-image__body-text, -.backup-phrase__body-text { +.backup-phrase__body-text, +.buy-ether__body-text { color: #1B344D; font-size: 16px; line-height: 23px; font-family: Montserrat UltraLight; } +.buy-ether__small-body-text { + font-family: Montserrat UltraLight; + height: 14px; + color: #757575; + font-size: 12px; + line-height: 14px; +} + .unique-image__body-text { width: 335px; } @@ -325,6 +335,89 @@ button.backup-phrase__confirm-seed-option:hover { line-height: 23px; margin-left: 22px; } + +.buy-ether__content-wrapper { + display: flex; + flex-flow: column nowrap; + margin-top: 31px; +} + +.buy-ether__content-headline { + color: #1B344D; + font-family: Montserrat Light; + font-size: 18px; + line-height: 23px; +} + +.buy-ether__content { + margin-top: 12px; + display: flex; + flex-flow: row nowrap; +} + +.buy-ether__side-panel { + display: flex; + flex-flow: column nowrap; +} + +.buy-ether__side-panel-item { + display: flex; + flex-flow: row nowrap; + align-items: center; + padding: 20px 0; + color: #9B9B9B; + font-family: Montserrat Light; + font-size: 14px; + line-height: 18px; + cursor: pointer; + min-width: 140px; + border-bottom: 1px solid #CDCDCD; +} + +.buy-ether__side-panel-item--selected { + position: relative; + color: #1B344D; +} + +.buy-ether__side-panel-item-name { + flex: 1 0 auto; + padding-right: 13px; +} + +.buy-ether__action-content { + margin-left: 34px; +} + +.buy-ether__buttons { + display: flex; + flex-flow: row nowrap; + align-items: center; +} + +.buy-ether__button-separator-text { + font-size: 20px; + line-height: 26px; + font-family: Montserrat Light; + margin: 35px 0 14px 30px; + display: flex; + flex-flow: column nowrap; + justify-content: center; +} + +.buy-ether__faq-link { + margin-top: 26px; + color: #1B344D !important; + font-size: 14px !important; + line-height: 18px !important; + font-family: Montserrat UltraLight !important; +} + +.buy-ether__action-content-wrapper { + width: 360px; + display: flex; + flex-flow: column nowrap; +} + .first-time-flow__input { width: 350px; font-size: 18px; @@ -363,6 +456,23 @@ button.first-time-flow__button:hover { background-color: rgba(247, 134, 28, 0.9); } +.first-time-flow__button--tertiary { + height: 54px; + width: 198px; + box-shadow: none; + color: #1B344D; + font-size: 20px; + line-height: 26px; + font-family: Montserrat Light; + text-align: center; + margin: 35px 0 14px; + background-color: transparent; +} + +button.first-time-flow__button--tertiary:hover { + transform: scale(1); +} + .first-time-flow__link { color: #1B344D; font-size: 18px; @@ -406,4 +516,13 @@ button.first-time-flow__button:hover { line-height: 26px; text-align: center; font-family: Montserrat UltraLight; -} \ No newline at end of file +} + +.shapeshift-logo { + background: url(''); + width: 161px; + height: 84px; + background-size: cover; + background-repeat: no-repeat; + background-position: 50%; +} diff --git a/mascara/src/app/first-time/index.js b/mascara/src/app/first-time/index.js index 1ba6ed28c..3fcfd8dc5 100644 --- a/mascara/src/app/first-time/index.js +++ b/mascara/src/app/first-time/index.js @@ -5,6 +5,7 @@ import UniqueImageScreen from './unique-image-screen' import NoticeScreen from './notice-screen' import BackupPhraseScreen from './backup-phrase-screen' import ImportAccountScreen from './import-account-screen' +import BuyEtherScreen from './buy-ether-screen' class FirstTimeFlow extends Component { @@ -45,7 +46,7 @@ class FirstTimeFlow extends Component { const {isInitialized, seedWords, noActiveNotices} = this.props; const {SCREEN_TYPE} = FirstTimeFlow - // return SCREEN_TYPE.IMPORT_ACCOUNT + return SCREEN_TYPE.BUY_ETHER if (!isInitialized) { return SCREEN_TYPE.CREATE_PASSWORD @@ -96,6 +97,12 @@ class FirstTimeFlow extends Component { next={() => this.setScreenType(SCREEN_TYPE.BUY_ETHER)} /> ) + case SCREEN_TYPE.BUY_ETHER: + return ( + + ) default: return