Merge branch 'develop' into fix-gas-slider

This commit is contained in:
Daniel Ternyak 2018-07-06 14:07:26 -05:00 committed by GitHub
commit 6cc8cafa64
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
33 changed files with 510 additions and 325 deletions

View File

@ -0,0 +1,125 @@
import shapeshift, { SHAPESHIFT_BASE_URL } from './shapeshift';
describe('ShapeShift service', () => {
beforeEach(() => {
(global as any).fetch = jest.fn().mockImplementation(
(url: string) =>
new Promise(resolve => {
const returnValues = {
[`${SHAPESHIFT_BASE_URL}/marketinfo`]: {
status: 200,
json: () => [
{
limit: 1,
maxLimit: 2,
min: 1,
minerFee: 2,
pair: 'BTC_ETH',
rate: '1.0'
},
{
limit: 1,
maxLimit: 2,
min: 1,
minerFee: 2,
pair: 'ETH_BTC',
rate: '1.0'
}
]
},
[`${SHAPESHIFT_BASE_URL}/getcoins`]: {
status: 200,
json: () => ({
BTC: {
name: 'Bitcoin',
symbol: 'BTC',
image: '',
imageSmall: '',
status: 'available',
minerFee: 1
},
ETH: {
name: 'Ethereum',
symbol: 'ETH',
image: '',
imageSmall: '',
status: 'available',
minerFee: 1
},
XMR: {
name: 'Monero',
symbol: 'XMR',
image: '',
imageSmall: '',
status: 'unavailable',
minerFee: 1
}
})
}
};
resolve(returnValues[url]);
})
);
});
it('provides a collection of all available and unavailable coins and tokens', async done => {
const rates = await shapeshift.getAllRates();
expect(rates).toEqual({
BTCETH: {
id: 'BTCETH',
rate: '1.0',
limit: 1,
min: 1,
options: [
{
id: 'BTC',
image: '',
name: 'Bitcoin',
status: 'available'
},
{
id: 'ETH',
image: '',
name: 'Ethereum',
status: 'available'
}
]
},
ETHBTC: {
id: 'ETHBTC',
rate: '1.0',
limit: 1,
min: 1,
options: [
{
id: 'ETH',
image: '',
name: 'Ethereum',
status: 'available'
},
{
id: 'BTC',
image: '',
name: 'Bitcoin',
status: 'available'
}
]
},
__XMR: {
id: '__XMR',
limit: 0,
min: 0,
options: [
{
id: 'XMR',
image: '',
name: 'Monero',
status: 'unavailable'
}
]
}
});
done();
});
});

View File

@ -1,9 +1,12 @@
import flatten from 'lodash/flatten';
import uniqBy from 'lodash/uniqBy';
import { checkHttpStatus, parseJSON } from 'api/utils'; import { checkHttpStatus, parseJSON } from 'api/utils';
const SHAPESHIFT_API_KEY = export const SHAPESHIFT_API_KEY =
'8abde0f70ca69d5851702d57b10305705d7333e93263124cc2a2649dab7ff9cf86401fc8de7677e8edcd0e7f1eed5270b1b49be8806937ef95d64839e319e6d9'; '8abde0f70ca69d5851702d57b10305705d7333e93263124cc2a2649dab7ff9cf86401fc8de7677e8edcd0e7f1eed5270b1b49be8806937ef95d64839e319e6d9';
const SHAPESHIFT_BASE_URL = 'https://shapeshift.io'; export const SHAPESHIFT_BASE_URL = 'https://shapeshift.io';
export const SHAPESHIFT_TOKEN_WHITELIST = [ export const SHAPESHIFT_TOKEN_WHITELIST = [
'OMG', 'OMG',
@ -26,7 +29,7 @@ export const SHAPESHIFT_TOKEN_WHITELIST = [
'TRST', 'TRST',
'GUP' 'GUP'
]; ];
export const SHAPESHIFT_WHITELIST = [...SHAPESHIFT_TOKEN_WHITELIST, 'ETH', 'ETC', 'BTC']; export const SHAPESHIFT_WHITELIST = [...SHAPESHIFT_TOKEN_WHITELIST, 'ETH', 'ETC', 'BTC', 'XMR'];
interface IPairData { interface IPairData {
limit: number; limit: number;
@ -66,6 +69,29 @@ interface TokenMap {
}; };
} }
interface ShapeshiftCoinInfo {
image: string;
imageSmall: string;
minerFee: number;
name: string;
status: string;
symbol: string;
}
interface ShapeshiftCoinInfoMap {
[id: string]: ShapeshiftCoinInfo;
}
interface ShapeshiftOption {
id?: string;
status?: string;
image?: string;
}
interface ShapeshiftOptionMap {
[symbol: string]: ShapeshiftOption;
}
class ShapeshiftService { class ShapeshiftService {
public whitelist = SHAPESHIFT_WHITELIST; public whitelist = SHAPESHIFT_WHITELIST;
private url = SHAPESHIFT_BASE_URL; private url = SHAPESHIFT_BASE_URL;
@ -73,6 +99,8 @@ class ShapeshiftService {
private postHeaders = { private postHeaders = {
'Content-Type': 'application/json' 'Content-Type': 'application/json'
}; };
private supportedCoinsAndTokens: ShapeshiftCoinInfoMap = {};
private fetchedSupportedCoinsAndTokens = false;
public checkStatus(address: string) { public checkStatus(address: string) {
return fetch(`${this.url}/txStat/${address}`) return fetch(`${this.url}/txStat/${address}`)
@ -118,19 +146,76 @@ class ShapeshiftService {
public getAllRates = async () => { public getAllRates = async () => {
const marketInfo = await this.getMarketInfo(); const marketInfo = await this.getMarketInfo();
const pairRates = await this.filterPairs(marketInfo); const pairRates = this.filterPairs(marketInfo);
const checkAvl = await this.checkAvl(pairRates); const checkAvl = await this.checkAvl(pairRates);
const mappedRates = this.mapMarketInfo(checkAvl); const mappedRates = this.mapMarketInfo(checkAvl);
return mappedRates; const allRates = this.addUnavailableCoinsAndTokens(mappedRates);
return allRates;
};
public addUnavailableCoinsAndTokens = (availableCoinsAndTokens: TokenMap) => {
if (this.fetchedSupportedCoinsAndTokens) {
/** @desc Create a hash for efficiently checking which tokens are currently available. */
const allOptions = flatten(
Object.values(availableCoinsAndTokens).map(({ options }) => options)
);
const availableOptions: ShapeshiftOptionMap = uniqBy(allOptions, 'id').reduce(
(prev: ShapeshiftOptionMap, next) => {
prev[next.id] = next;
return prev;
},
{}
);
const unavailableCoinsAndTokens = this.whitelist
.map(token => {
/** @desc ShapeShift claims support for the token and it is available. */
const availableCoinOrToken = availableOptions[token];
if (availableCoinOrToken) {
return null;
}
/** @desc ShapeShift claims support for the token, but it is unavailable. */
const supportedCoinOrToken = this.supportedCoinsAndTokens[token];
if (supportedCoinOrToken) {
const { symbol: id, image, name, status } = supportedCoinOrToken;
return {
/** @desc Preface the false id with '__' to differentiate from actual pairs. */
id: `__${id}`,
limit: 0,
min: 0,
options: [{ id, image, name, status }]
};
}
/** @desc We claim support for the coin or token, but ShapeShift doesn't. */
return null;
})
.reduce((prev: ShapeshiftOptionMap, next) => {
if (next) {
prev[next.id] = next;
return prev;
}
return prev;
}, {});
return { ...availableCoinsAndTokens, ...unavailableCoinsAndTokens };
}
return availableCoinsAndTokens;
}; };
private filterPairs(marketInfo: ShapeshiftMarketInfo[]) { private filterPairs(marketInfo: ShapeshiftMarketInfo[]) {
return marketInfo.filter(obj => { return marketInfo.filter(obj => {
const { pair } = obj; const { pair } = obj;
const pairArr = pair.split('_'); const pairArr = pair.split('_');
return this.whitelist.includes(pairArr[0]) && this.whitelist.includes(pairArr[1]) return this.whitelist.includes(pairArr[0]) && this.whitelist.includes(pairArr[1]);
? true
: false;
}); });
} }
@ -165,7 +250,13 @@ class ShapeshiftService {
private getAvlCoins() { private getAvlCoins() {
return fetch(`${this.url}/getcoins`) return fetch(`${this.url}/getcoins`)
.then(checkHttpStatus) .then(checkHttpStatus)
.then(parseJSON); .then(parseJSON)
.then(supportedCoinsAndTokens => {
this.supportedCoinsAndTokens = supportedCoinsAndTokens;
this.fetchedSupportedCoinsAndTokens = true;
return supportedCoinsAndTokens;
});
} }
private getMarketInfo() { private getMarketInfo() {

View File

@ -1,53 +0,0 @@
@import 'common/sass/variables';
@import 'common/sass/mixins';
.AppAlpha {
@include cover-message;
background: color(brand-info);
left: $electron-sidebar-width - 1;
&-content {
h2 {
text-align: center;
}
p {
text-align: justify;
}
&-btn {
display: block;
width: 100%;
max-width: 280px;
margin: 40px auto 0;
border: none;
padding: 0;
transition: $transition;
height: 60px;
line-height: 60px;
font-size: 22px;
background: #fff;
color: #333;
opacity: 0.96;
border-radius: 4px;
&:hover {
opacity: 1;
}
}
}
// Fade out
&.is-fading {
pointer-events: none;
opacity: 0;
background: color(control-bg);
transition: all 500ms ease 400ms;
.AppAlpha-content {
opacity: 0;
transform: translateY(15px);
transition: all 500ms ease;
}
}
}

View File

@ -1,73 +0,0 @@
import React from 'react';
import moment from 'moment';
import { discordURL, APP_ALPHA_EXPIRATION } from 'config';
import { NewTabLink } from 'components/ui';
import './AlphaNotice.scss';
interface State {
isFading: boolean;
isClosed: boolean;
}
let hasAcknowledged = false;
export default class AppAlphaNotice extends React.PureComponent<{}, State> {
public state = {
isFading: false,
isClosed: hasAcknowledged
};
public render() {
if (this.state.isClosed) {
return null;
}
const isFading = this.state.isFading ? 'is-fading' : '';
const expDate = moment(APP_ALPHA_EXPIRATION).format('MMMM Do, YYYY');
return (
<div className={`AppAlpha ${isFading}`}>
<div className="AppAlpha-content">
<h2>Welcome to the MyCrypto Desktop App Alpha</h2>
<p>
Thank you for testing out the new MyCrypto desktop app. This is an early release to be
tested by the community before a full launch. We recommend continuing to use the
production site for large or otherwise important transactions.
</p>
<p>
Because this is for testing purposes only,{' '}
<strong>this build of the app will only be accessible until {expDate}</strong>. Youll
then be required to update the application to continue using it.
</p>
<p>
Feedback and bug reports are greatly appreciated. You can file issues on our{' '}
<NewTabLink href="https://github.com/MyCryptoHQ/MyCrypto/issues">
GitHub repository
</NewTabLink>{' '}
or join our <NewTabLink href={discordURL}>Discord server</NewTabLink> to discuss the
app.
</p>
<p>
<b>
For critical reports & vulnerabilities, please use{' '}
<NewTabLink href="https://hackerone.com/MyCrypto">HackerOne</NewTabLink>.
</b>
</p>
<button className="AppAlpha-content-btn is-continue" onClick={this.doContinue}>
Continue to the Alpha
</button>
</div>
</div>
);
}
private doContinue = () => {
hasAcknowledged = true;
this.setState({ isFading: true });
setTimeout(() => {
this.setState({ isClosed: true });
}, 1000);
};
}

View File

@ -1,44 +0,0 @@
@import 'common/sass/variables';
@import 'common/sass/mixins';
.AppExpired {
@include cover-message;
background: color(brand-danger);
display: flex;
align-items: center;
&-content {
padding-bottom: 60px;
h2 {
text-align: center;
}
p {
text-align: justify;
}
&-btn {
display: block;
width: 100%;
max-width: 280px;
margin: 40px auto 0;
text-align: center;
border: none;
padding: 0;
transition: $transition;
height: 60px;
line-height: 60px;
font-size: 22px;
background: #fff;
color: #333;
border-radius: 4px;
text-shadow: none;
opacity: 0.95;
&:hover {
opacity: 1;
}
}
}
}

View File

@ -1,26 +0,0 @@
import React from 'react';
import { NewTabLink } from 'components/ui';
import './AppExpired.scss';
const AppExpired: React.SFC<{}> = () => (
<div className="AppExpired">
<div className="AppExpired-content">
<h2>Your Alpha Build Has Expired</h2>
<p>
To ensure the safety of your funds, we are expiring alpha builds one month after release and
requiring users to update. All you have to do is download a new build from our GitHub, and
you can continue to use the app. Sorry for the hassle!
</p>
<NewTabLink
href="https://github.com/MyCryptoHQ/MyCrypto/releases/latest"
className="AppExpired-content-btn"
>
Download a New Build
</NewTabLink>
</div>
</div>
);
export default AppExpired;

View File

@ -1,15 +0,0 @@
import React from 'react';
import { APP_ALPHA_EXPIRATION } from 'config';
import AlphaNotice from './AlphaNotice';
import AppExpired from './AppExpired';
const AppAlphaNotice: React.SFC<{}> = () => {
if (APP_ALPHA_EXPIRATION < Date.now()) {
return <AppExpired />;
} else {
return <AlphaNotice />;
}
};
export default AppAlphaNotice;

View File

@ -44,7 +44,6 @@ class ElectronNav extends React.Component<Props, State> {
> >
<div className="ElectronNav-branding"> <div className="ElectronNav-branding">
<div className="ElectronNav-branding-logo" onClick={this.toggleTheme} /> <div className="ElectronNav-branding-logo" onClick={this.toggleTheme} />
<div className="ElectronNav-branding-beta">Alpha Release</div>
</div> </div>
<ul className="ElectronNav-links"> <ul className="ElectronNav-links">

View File

@ -24,4 +24,3 @@ export { default as ParityQrSigner } from './ParityQrSigner';
export { default as ElectronNav } from './ElectronNav'; export { default as ElectronNav } from './ElectronNav';
export { default as AddressBookTable } from './AddressBookTable'; export { default as AddressBookTable } from './AddressBookTable';
export { default as Errorable } from './Errorable'; export { default as Errorable } from './Errorable';
export { default as AppAlphaNotice } from './AppAlphaNotice';

View File

@ -87,7 +87,7 @@ class SwapDropdown extends PureComponent<Props, State> {
key={opt.name} key={opt.name}
option={opt} option={opt}
isMain={false} isMain={false}
isDisabled={opt.name === disabledOption} isDisabled={opt.name === disabledOption || opt.status === 'unavailable'}
onChange={this.handleChange} onChange={this.handleChange}
/> />
))} ))}
@ -140,6 +140,23 @@ class SwapDropdown extends PureComponent<Props, State> {
(opt1, opt2) => (opt1.id.toLowerCase() > opt2.id.toLowerCase() ? 1 : -1) (opt1, opt2) => (opt1.id.toLowerCase() > opt2.id.toLowerCase() ? 1 : -1)
); );
// Sort unavailable options last
otherOptions = otherOptions.sort((opt1, opt2) => {
if (opt1.status === 'available' && opt2.status === 'unavailable') {
return -1;
}
if (opt1.status === 'available' && opt2.status === 'available') {
return 0;
}
if (opt1.status === 'unavailable' && opt2.status === 'available') {
return 1;
}
return 0;
});
this.setState({ mainOptions, otherOptions }); this.setState({ mainOptions, otherOptions });
} }
} }

View File

@ -0,0 +1,35 @@
@import 'common/sass/variables';
@import 'common/sass/mixins';
.Warning {
display: flex;
align-items: center;
border-top: 2px solid $brand-danger;
padding: $space-sm;
font-size: $font-size-base;
line-height: 1.5;
font-weight: 500;
box-shadow: 0 1px 1px 1px rgba(#000, 0.12);
margin-bottom: $space;
&.highlighted {
background-color: lighten($brand-danger, 30%);
}
&-icon {
display: flex;
flex-direction: column;
justify-content: center;
width: 32px;
margin-left: $space * 0.4;
margin-right: $space * 0.8;
text-align: center;
font-size: 32px;
color: $brand-danger;
}
&-content {
display: flex;
flex-direction: column;
padding: 0 $space;
}
}

View File

@ -0,0 +1,22 @@
import React from 'react';
import './Warning.scss';
interface WarningProps {
highlighted?: boolean;
}
const Warning: React.SFC<WarningProps> = ({ highlighted, children }) => {
const className = `Warning ${highlighted ? 'highlighted' : ''}`;
return (
<section className={className}>
<section className="Warning-icon">
<i className="fa fa-exclamation-triangle" />
</section>
<section className="Warning-content">{children}</section>
</section>
);
};
export default Warning;

View File

@ -16,5 +16,6 @@ export { default as TextArea } from './TextArea';
export { default as Address } from './Address'; export { default as Address } from './Address';
export { default as CodeBlock } from './CodeBlock'; export { default as CodeBlock } from './CodeBlock';
export { default as Toggle } from './Toggle'; export { default as Toggle } from './Toggle';
export { default as Warning } from './Warning';
export * from './Expandable'; export * from './Expandable';
export * from './InlineSpinner'; export * from './InlineSpinner';

View File

@ -1,6 +1,6 @@
import { BTCTxExplorer, ETHTxExplorer } from './data'; import { BTCTxExplorer, ETHTxExplorer } from './data';
export type WhitelistedCoins = 'BTC' | 'REP' | 'ETH'; export type WhitelistedCoins = 'BTC' | 'REP' | 'ETH' | 'XMR';
const serverURL = 'https://bity.myetherapi.com'; const serverURL = 'https://bity.myetherapi.com';
const bityURL = 'https://bity.com/api'; const bityURL = 'https://bity.com/api';
const BTCMin = 0.01; const BTCMin = 0.01;
@ -11,7 +11,8 @@ const BTCMax = 3;
// value = percent higher/lower than 0.01 BTC worth // value = percent higher/lower than 0.01 BTC worth
const buffers = { const buffers = {
ETH: 0.1, ETH: 0.1,
REP: 0.2 REP: 0.2,
XMR: 0.3
}; };
// rate must be BTC[KIND] // rate must be BTC[KIND]

View File

@ -12,12 +12,6 @@ export const discordURL = 'https://discord.gg/VSaTXEA';
export const VERSION = packageJson.version; export const VERSION = packageJson.version;
export const N_FACTOR = 8192; export const N_FACTOR = 8192;
// Bricks the app once this date has been exceeded. Remember to update these 2
// whenever making a new app release.
// It is currently set to: Wednesday, July 25, 2018 12:00:00 AM (GMT)
// TODO: Remove me once app alpha / release candidates are done
export const APP_ALPHA_EXPIRATION = 1532476800000;
// Displays at the top of the site, make message empty string to remove. // Displays at the top of the site, make message empty string to remove.
// Type can be primary, warning, danger, success, info, or blank for grey. // Type can be primary, warning, danger, success, info, or blank for grey.
// Message must be a JSX element if you want to use HTML. // Message must be a JSX element if you want to use HTML.
@ -44,7 +38,9 @@ export const etherChainExplorerInst = makeExplorer({
export const donationAddressMap = { export const donationAddressMap = {
BTC: '32oirLEzZRhi33RCXDF9WHJjEb8RsrSss3', BTC: '32oirLEzZRhi33RCXDF9WHJjEb8RsrSss3',
ETH: '0x4bbeEB066eD09B7AEd07bF39EEe0460DFa261520', ETH: '0x4bbeEB066eD09B7AEd07bF39EEe0460DFa261520',
REP: '0x4bbeEB066eD09B7AEd07bF39EEe0460DFa261520' REP: '0x4bbeEB066eD09B7AEd07bF39EEe0460DFa261520',
XMR:
'4GdoN7NCTi8a5gZug7PrwZNKjvHFmKeV11L6pNJPgj5QNEHsN6eeX3DaAQFwZ1ufD4LYCZKArktt113W7QjWvQ7CW7F7tDFvS511SNfZV7'
}; };
export const gasEstimateCacheTime = 60000; export const gasEstimateCacheTime = 60000;

View File

@ -3328,5 +3328,32 @@
"address": "0xAc709FcB44a43c35F0DA4e3163b117A17F3770f5", "address": "0xAc709FcB44a43c35F0DA4e3163b117A17F3770f5",
"symbol": "ARC", "symbol": "ARC",
"decimal": 18 "decimal": 18
},
{
"address": "0xDF2C7238198Ad8B389666574f2d8bc411A4b7428",
"symbol": "MFT",
"decimal": 18
},
{
"address": "0xc92d6e3e64302c59d734f3292e2a13a13d7e1817",
"symbol": "FXC",
"decimal": 8
},
{
"address": "0x4f3afec4e5a3f2a6a1a411def7d7dfe50ee057bf",
"symbol": "DGX 2.0",
"decimal": 9
},
{
"address": "0xd9a12cde03a86e800496469858de8581d3a5353d",
"symbol": "YUP",
"decimal": 18
},
{
"address": "0x6aEDbF8dFF31437220dF351950Ba2a3362168d1b",
"symbol": "DGS",
"decimal": 8
} }
] ]

View File

@ -1,27 +0,0 @@
@import 'common/sass/variables';
@import 'common/sass/mixins';
.WelcomeSlide {
&-alert {
display: flex;
border-top: 2px solid color(brand-danger);
padding: $space-sm;
font-size: $font-size-base;
line-height: 1.5;
font-weight: 500;
box-shadow: 0 1px 1px 1px rgba(#000, 0.12);
margin-bottom: $space;
&-icon {
display: flex;
flex-direction: column;
justify-content: center;
width: 32px;
margin-left: $space * 0.4;
margin-right: $space * 0.8;
text-align: center;
font-size: 32px;
color: color(brand-danger);
}
}
}

View File

@ -2,31 +2,20 @@ import React from 'react';
import translate from 'translations'; import translate from 'translations';
import onboardIconOne from 'assets/images/onboarding/slide-01.svg'; import onboardIconOne from 'assets/images/onboarding/slide-01.svg';
import { Warning } from 'components/ui';
import OnboardSlide from './OnboardSlide'; import OnboardSlide from './OnboardSlide';
import './WelcomeSlide.scss';
const WelcomeSlide = () => { const WelcomeSlide = () => {
const header = translate('ONBOARD_WELCOME_TITLE'); const header = translate('ONBOARD_WELCOME_TITLE');
const subheader = <small>{translate('ONBOARD_WELCOME_CONTENT__3')}</small>; const subheader = <small>{translate('ONBOARD_WELCOME_CONTENT__3')}</small>;
const content = ( const content = (
<div> <div>
<div className="WelcomeSlide-alert"> <Warning>
<div className="WelcomeSlide-alert-icon"> {translate('ONBOARD_WELCOME_CONTENT__1')}
<i className="fa fa-exclamation-triangle" /> {translate('ONBOARD_WELCOME_CONTENT__2')}
</div> </Warning>
<span> <Warning>{translate('ONBOARD_WELCOME_CONTENT__8')}</Warning>
{translate('ONBOARD_WELCOME_CONTENT__1')}
{translate('ONBOARD_WELCOME_CONTENT__2')}
</span>
</div>
<div className="WelcomeSlide-alert">
<div className="WelcomeSlide-alert-icon">
<i className="fa fa-exclamation-triangle" />
</div>
{translate('ONBOARD_WELCOME_CONTENT__8')}
</div>
<h5>{translate('ONBOARD_WELCOME_CONTENT__4')}</h5> <h5>{translate('ONBOARD_WELCOME_CONTENT__4')}</h5>
<ul> <ul>
<li>{translate('ONBOARD_WELCOME_CONTENT__5')}</li> <li>{translate('ONBOARD_WELCOME_CONTENT__5')}</li>

View File

@ -3,7 +3,7 @@ import { connect } from 'react-redux';
import { AppState } from 'features/reducers'; import { AppState } from 'features/reducers';
import { getOffline } from 'features/config'; import { getOffline } from 'features/config';
import { ElectronNav, AppAlphaNotice } from 'components'; import { ElectronNav } from 'components';
import OfflineTab from './OfflineTab'; import OfflineTab from './OfflineTab';
import Notifications from './Notifications'; import Notifications from './Notifications';
import './ElectronTemplate.scss'; import './ElectronTemplate.scss';
@ -33,7 +33,6 @@ class ElectronTemplate extends Component<Props, {}> {
{isUnavailableOffline && isOffline ? <OfflineTab /> : children} {isUnavailableOffline && isOffline ? <OfflineTab /> : children}
</div> </div>
<Notifications /> <Notifications />
<AppAlphaNotice />
</div> </div>
<div className="ElectronTemplate-draggable" /> <div className="ElectronTemplate-draggable" />
</div> </div>

View File

@ -40,12 +40,13 @@
} }
&-input { &-input {
color: #fff;
display: inline-block; display: inline-block;
width: 16%; width: 16%;
min-width: 3.5rem; min-width: 3.5rem;
height: 2rem; height: 2rem;
padding: 1rem; padding: 1rem;
margin: .5rem auto; margin: 0.5rem auto;
font-size: 1rem; font-size: 1rem;
margin-right: $space; margin-right: $space;
text-align: right; text-align: right;

View File

@ -27,6 +27,8 @@ interface ReduxStateProps {
bityOrderStatus: string | null; bityOrderStatus: string | null;
shapeshiftOrderStatus: string | null; shapeshiftOrderStatus: string | null;
outputTx: any; outputTx: any;
paymentId: string | null;
xmrPaymentAddress: string | null;
} }
interface ReduxActionProps { interface ReduxActionProps {
@ -68,6 +70,8 @@ export default class PartThree extends PureComponent<ReduxActionProps & ReduxSta
shapeshiftOrderStatus, shapeshiftOrderStatus,
destinationAddress, destinationAddress,
outputTx, outputTx,
paymentId,
xmrPaymentAddress,
// ACTIONS // ACTIONS
showNotificationWithComponent showNotificationWithComponent
} = this.props; } = this.props;
@ -85,7 +89,9 @@ export default class PartThree extends PureComponent<ReduxActionProps & ReduxSta
const PaymentInfoProps = { const PaymentInfoProps = {
origin, origin,
paymentAddress paymentAddress,
paymentId,
xmrPaymentAddress
}; };
const BitcoinQRProps = { const BitcoinQRProps = {

View File

@ -1,18 +1,31 @@
@import "common/sass/variables"; @import 'common/sass/variables';
@import "common/sass/mixins"; @import 'common/sass/mixins';
.SwapPayment { .SwapPayment {
text-align: center; > * {
margin-bottom: $space; margin: $space auto 0;
width: 100%;
max-width: 620px;
}
h2 {
text-align: center;
& > input {
margin-top: $space;
}
}
h4 {
margin: 0.5rem 0;
}
&-address { &-address {
@include mono;
display: block; display: block;
margin: $space auto 0;
max-width: 620px;
width: 100%;
font-size: $font-size-medium; font-size: $font-size-medium;
text-align: center; text-align: center;
@include mono;
} }
@media screen and (max-width: $screen-sm) { @media screen and (max-width: $screen-sm) {
@ -20,4 +33,16 @@
font-size: $font-size-base; font-size: $font-size-base;
} }
} }
&-payment-id {
h2 {
font-weight: bolder;
text-align: center;
}
&-link {
font-size: $font-size-small;
text-align: right;
}
}
} }

View File

@ -2,19 +2,47 @@ import React, { PureComponent } from 'react';
import translate from 'translations'; import translate from 'translations';
import { SwapInput } from 'features/swap/types'; import { SwapInput } from 'features/swap/types';
import { Input } from 'components/ui'; import { Input, Warning } from 'components/ui';
import './PaymentInfo.scss'; import './PaymentInfo.scss';
export interface Props { export interface Props {
origin: SwapInput; origin: SwapInput;
paymentAddress: string | null; paymentAddress: string | null;
/**
* @desc
* For XMR swaps, the "deposit" property in the response
* actually refers to the "paymentId", not the payment address.
*/
paymentId: string | null;
/**
* @desc
* For XMR swap, the actual payment address is the "sAddress"
* property in the response.
*/
xmrPaymentAddress: string | null;
} }
export default class PaymentInfo extends PureComponent<Props, {}> { export default class PaymentInfo extends PureComponent<Props, {}> {
public render() { public render() {
const { origin } = this.props; const { origin, paymentAddress, paymentId, xmrPaymentAddress } = this.props;
const isXMRSwap = origin.label === 'XMR';
const actualPaymentAddress = isXMRSwap ? xmrPaymentAddress : paymentAddress;
return ( return (
<section className="SwapPayment"> <section className="SwapPayment">
{isXMRSwap && (
<section className="SwapPayment-payment-id">
<h2>
{translate('USING_PAYMENT_ID')}
<Input
className="SwapPayment-address"
isValid={!!paymentId}
value={paymentId || undefined}
disabled={true}
/>
</h2>
</section>
)}
<h2> <h2>
{translate('SWAP_SEND_TO', { {translate('SWAP_SEND_TO', {
$origin_amount: origin.amount.toString(), $origin_amount: origin.amount.toString(),
@ -22,11 +50,16 @@ export default class PaymentInfo extends PureComponent<Props, {}> {
})} })}
<Input <Input
className="SwapPayment-address" className="SwapPayment-address"
isValid={!!this.props.paymentAddress} isValid={!!actualPaymentAddress}
value={this.props.paymentAddress || undefined} value={actualPaymentAddress || undefined}
disabled={true} disabled={true}
/> />
</h2> </h2>
{isXMRSwap && (
<Warning highlighted={true}>
<h4>{translate('PAYMENT_ID_WARNING')}</h4>
</Warning>
)}
</section> </section>
); );
} }

View File

@ -1,8 +1,8 @@
import React, { PureComponent } from 'react'; import React, { PureComponent } from 'react';
import { donationAddressMap } from 'config'; import { donationAddressMap, WhitelistedCoins } from 'config';
import translate, { translateRaw } from 'translations'; import translate, { translateRaw } from 'translations';
import { isValidBTCAddress, isValidETHAddress } from 'libs/validators'; import { isValidBTCAddress, isValidETHAddress, isValidXMRAddress } from 'libs/validators';
import { combineAndUpper } from 'utils/formatters'; import { combineAndUpper } from 'utils/formatters';
import { SwapInput } from 'features/swap/types'; import { SwapInput } from 'features/swap/types';
import { import {
@ -18,7 +18,7 @@ import './ReceivingAddress.scss';
export interface StateProps { export interface StateProps {
origin: SwapInput; origin: SwapInput;
destinationId: keyof typeof donationAddressMap; destinationId: WhitelistedCoins;
isPostingOrder: boolean; isPostingOrder: boolean;
destinationAddress: string; destinationAddress: string;
destinationKind: number; destinationKind: number;
@ -62,13 +62,22 @@ export default class ReceivingAddress extends PureComponent<StateProps & ActionP
public render() { public render() {
const { destinationId, destinationAddress, isPostingOrder } = this.props; const { destinationId, destinationAddress, isPostingOrder } = this.props;
let validAddress;
// TODO - find better pattern here once currencies move beyond BTC, ETH, REP const addressValidators: { [coinOrToken: string]: (address: string) => boolean } = {
if (destinationId === 'BTC') { BTC: isValidBTCAddress,
validAddress = isValidBTCAddress(destinationAddress); XMR: isValidXMRAddress,
} else { ETH: isValidETHAddress
validAddress = isValidETHAddress(destinationAddress); };
} // If there is no matching validator for the ID, assume it's a token and use ETH.
const addressValidator = addressValidators[destinationId] || addressValidators.ETH;
const validAddress = addressValidator(destinationAddress);
const placeholders: { [coinOrToken: string]: string } = {
BTC: donationAddressMap.BTC,
XMR: donationAddressMap.XMR,
ETH: donationAddressMap.ETH
};
const placeholder = placeholders[destinationId] || donationAddressMap.ETH;
return ( return (
<section className="SwapAddress block"> <section className="SwapAddress block">
@ -85,11 +94,7 @@ export default class ReceivingAddress extends PureComponent<StateProps & ActionP
type="text" type="text"
value={destinationAddress} value={destinationAddress}
onChange={this.onChangeDestinationAddress} onChange={this.onChangeDestinationAddress}
placeholder={ placeholder={placeholder}
destinationId === 'BTC'
? donationAddressMap[destinationId]
: donationAddressMap.ETH
}
/> />
</label> </label>
</div> </div>

View File

@ -34,6 +34,8 @@ interface ReduxStateProps {
bityOrderStatus: string | null; bityOrderStatus: string | null;
shapeshiftOrderStatus: string | null; shapeshiftOrderStatus: string | null;
paymentAddress: string | null; paymentAddress: string | null;
paymentId: string | null;
xmrPaymentAddress: string | null;
isOffline: boolean; isOffline: boolean;
} }
@ -77,6 +79,8 @@ class Swap extends Component<ReduxActionProps & ReduxStateProps & RouteComponent
shapeshiftOrderStatus, shapeshiftOrderStatus,
isPostingOrder, isPostingOrder,
outputTx, outputTx,
paymentId,
xmrPaymentAddress,
// ACTIONS // ACTIONS
initSwap, initSwap,
restartSwap, restartSwap,
@ -159,7 +163,9 @@ class Swap extends Component<ReduxActionProps & ReduxStateProps & RouteComponent
stopPollShapeshiftOrderStatus, stopPollShapeshiftOrderStatus,
showNotificationWithComponent, showNotificationWithComponent,
destinationAddress, destinationAddress,
outputTx outputTx,
paymentId,
xmrPaymentAddress
}; };
const SupportProps = { const SupportProps = {
@ -220,6 +226,8 @@ function mapStateToProps(state: AppState) {
bityOrderStatus: state.swap.bityOrderStatus, bityOrderStatus: state.swap.bityOrderStatus,
shapeshiftOrderStatus: state.swap.shapeshiftOrderStatus, shapeshiftOrderStatus: state.swap.shapeshiftOrderStatus,
paymentAddress: state.swap.paymentAddress, paymentAddress: state.swap.paymentAddress,
paymentId: state.swap.paymentId,
xmrPaymentAddress: state.swap.xmrPaymentAddress,
isOffline: getOffline(state) isOffline: getOffline(state)
}; };
} }

View File

@ -234,7 +234,8 @@ describe('swap reducer', () => {
maxLimit: 7.04575258, maxLimit: 7.04575258,
apiPubKey: apiPubKey:
'0ca1ccd50b708a3f8c02327f0caeeece06d3ddc1b0ac749a987b453ee0f4a29bdb5da2e53bc35e57fb4bb7ae1f43c93bb098c3c4716375fc1001c55d8c94c160', '0ca1ccd50b708a3f8c02327f0caeeece06d3ddc1b0ac749a987b453ee0f4a29bdb5da2e53bc35e57fb4bb7ae1f43c93bb098c3c4716375fc1001c55d8c94c160',
minerFee: '1.05' minerFee: '1.05',
sAddress: '0x055ed77933388642fdn4px9v73j4fa3582d10c4'
}; };
const swapState = reducer.swapReducer( const swapState = reducer.swapReducer(
@ -254,8 +255,10 @@ describe('swap reducer', () => {
validFor: swapState.validFor, validFor: swapState.validFor,
orderTimestampCreatedISOString: swapState.orderTimestampCreatedISOString, orderTimestampCreatedISOString: swapState.orderTimestampCreatedISOString,
paymentAddress: mockedShapeshiftOrder.deposit, paymentAddress: mockedShapeshiftOrder.deposit,
paymentId: mockedShapeshiftOrder.deposit,
shapeshiftOrderStatus: 'no_deposits', shapeshiftOrderStatus: 'no_deposits',
orderId: mockedShapeshiftOrder.orderId orderId: mockedShapeshiftOrder.orderId,
xmrPaymentAddress: mockedShapeshiftOrder.sAddress
}); });
}); });

View File

@ -42,7 +42,9 @@ export const INITIAL_STATE: types.SwapState = {
paymentAddress: null, paymentAddress: null,
validFor: null, validFor: null,
orderId: null, orderId: null,
showLiteSend: false showLiteSend: false,
paymentId: null,
xmrPaymentAddress: null
}; };
export function swapReducer(state: types.SwapState = INITIAL_STATE, action: types.SwapAction) { export function swapReducer(state: types.SwapState = INITIAL_STATE, action: types.SwapAction) {
@ -69,22 +71,19 @@ export function swapReducer(state: types.SwapState = INITIAL_STATE, action: type
isFetchingRates: false isFetchingRates: false
}; };
case types.SwapActions.LOAD_SHAPESHIFT_RATES_SUCCEEDED: case types.SwapActions.LOAD_SHAPESHIFT_RATES_SUCCEEDED:
const {
entities: { providerRates: normalizedProviderRates, options: normalizedOptions }
} = normalize(action.payload, [providerRate]);
return { return {
...state, ...state,
shapeshiftRates: { shapeshiftRates: {
byId: normalize(action.payload, [providerRate]).entities.providerRates, byId: normalizedProviderRates,
allIds: allIds(normalize(action.payload, [providerRate]).entities.providerRates) allIds: allIds(normalizedProviderRates)
}, },
options: { options: {
byId: Object.assign( byId: { ...normalizedOptions, ...state.options.byId },
{}, allIds: [...allIds(normalizedOptions), ...state.options.allIds]
normalize(action.payload, [providerRate]).entities.options,
state.options.byId
),
allIds: [
...allIds(normalize(action.payload, [providerRate]).entities.options),
...state.options.allIds
]
}, },
isFetchingRates: false isFetchingRates: false
}; };
@ -151,8 +150,8 @@ export function swapReducer(state: types.SwapState = INITIAL_STATE, action: type
}; };
case types.SwapActions.SHAPESHIFT_ORDER_CREATE_SUCCEEDED: case types.SwapActions.SHAPESHIFT_ORDER_CREATE_SUCCEEDED:
const currDate = Date.now(); const currDate = Date.now();
const secondsRemaining = Math.floor((+new Date(action.payload.expiration) - currDate) / 1000); const secondsRemaining = Math.floor((+new Date(action.payload.expiration) - currDate) / 1000);
return { return {
...state, ...state,
shapeshiftOrder: { shapeshiftOrder: {
@ -166,7 +165,10 @@ export function swapReducer(state: types.SwapState = INITIAL_STATE, action: type
orderTimestampCreatedISOString: new Date(currDate).toISOString(), orderTimestampCreatedISOString: new Date(currDate).toISOString(),
paymentAddress: action.payload.deposit, paymentAddress: action.payload.deposit,
shapeshiftOrderStatus: 'no_deposits', shapeshiftOrderStatus: 'no_deposits',
orderId: action.payload.orderId orderId: action.payload.orderId,
// For XMR swaps
paymentId: action.payload.deposit,
xmrPaymentAddress: action.payload.sAddress
}; };
case types.SwapActions.BITY_ORDER_STATUS_SUCCEEDED: case types.SwapActions.BITY_ORDER_STATUS_SUCCEEDED:
return { return {

View File

@ -24,6 +24,18 @@ export interface SwapState {
validFor: number | null; validFor: number | null;
orderId: string | null; orderId: string | null;
showLiteSend: boolean; showLiteSend: boolean;
/**
* @desc
* For XMR swaps, the "deposit" property in the response
* actually refers to the "paymentId", not the payment address.
*/
paymentId: string | null;
/**
* @desc
* For XMR swap, the actual payment address is the "sAddress"
* property in the response.
*/
xmrPaymentAddress: string | null;
} }
export enum SwapActions { export enum SwapActions {
@ -208,6 +220,7 @@ export interface ShapeshiftOrderResponse {
quotedRate: string; quotedRate: string;
withdrawal: string; withdrawal: string;
withdrawalAmount: string; withdrawalAmount: string;
sAddress?: string;
} }
export interface ShapeshiftStatusResponse { export interface ShapeshiftStatusResponse {

View File

@ -59,6 +59,12 @@ export function isValidBTCAddress(address: string): boolean {
return WalletAddressValidator.validate(address, 'BTC'); return WalletAddressValidator.validate(address, 'BTC');
} }
export function isValidXMRAddress(address: string): boolean {
return !!address.match(
/4[0-9AB][123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz]{93}/
);
}
export function isValidHex(str: string): boolean { export function isValidHex(str: string): boolean {
if (str === '') { if (str === '') {
return true; return true;

View File

@ -660,6 +660,9 @@
"NETWORK_2": "network", "NETWORK_2": "network",
"PROVIDED_BY": "provided by", "PROVIDED_BY": "provided by",
"YOU_ARE_INTERACTING": "You are interacting with the", "YOU_ARE_INTERACTING": "You are interacting with the",
"USING_PAYMENT_ID": "Using the required Payment ID of:",
"PAYMENT_ID_WARNING": "Don't forget to send your XMR with the payment ID [[?]](https://getmonero.org/resources/moneropedia/paymentid.html) above, or you WILL lose your funds.",
"WHAT_IS_PAYMENT_ID": "what's a payment ID?",
"ANNOUNCEMENT_MESSAGE": "Welcome to the new MyCrypto. We hope you like it! If it's urgent and you need the old site, you can still use [MyCrypto Legacy](https://legacy.mycrypto.com)", "ANNOUNCEMENT_MESSAGE": "Welcome to the new MyCrypto. We hope you like it! If it's urgent and you need the old site, you can still use [MyCrypto Legacy](https://legacy.mycrypto.com)",
"U2F_NOT_SUPPORTED": "The U2F standard that hardware wallets use does not seem to be supported by your browser. Please try again using Google Chrome." "U2F_NOT_SUPPORTED": "The U2F standard that hardware wallets use does not seem to be supported by your browser. Please try again using Google Chrome."
} }

View File

@ -76,6 +76,7 @@ exports[`render snapshot 1`] = `
} }
outputTx={null} outputTx={null}
paymentAddress={null} paymentAddress={null}
paymentId={null}
provider="shapeshift" provider="shapeshift"
restartSwap={[Function]} restartSwap={[Function]}
secondsRemaining={null} secondsRemaining={null}
@ -100,5 +101,6 @@ exports[`render snapshot 1`] = `
stopPollBityOrderStatus={[Function]} stopPollBityOrderStatus={[Function]}
stopPollShapeshiftOrderStatus={[Function]} stopPollShapeshiftOrderStatus={[Function]}
swapProvider={[Function]} swapProvider={[Function]}
xmrPaymentAddress={null}
/> />
`; `;

2
static/robots.txt Normal file
View File

@ -0,0 +1,2 @@
User-agent: *
Sitemap: https://mycrypto.com/sitemap.xml

13
static/sitemap.xml Normal file
View File

@ -0,0 +1,13 @@
<?xml version="1.0" encoding="UTF-8"?>
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9">
<url><loc>https://mycrypto.com</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/account</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/generate</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/swap</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/contracts</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/ens</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/sign-and-verify-message</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/tx-status</loc><priority>0.5</priority></url>
<url><loc>https://mycrypto.com/pushTx</loc><priority>0.5</priority></url>
</urlset>