2018-06-06 15:37:03 -04:00
|
|
|
import React, { Fragment, PureComponent } from 'react';
|
2018-06-20 11:13:33 -04:00
|
|
|
import web3 from 'web3';
|
2018-06-13 15:04:06 -04:00
|
|
|
import { connect } from 'react-redux';
|
2018-08-01 16:00:15 -04:00
|
|
|
import { actions as accountActions, getDefaultAccount } from '../../reducers/accounts';
|
2018-07-09 17:53:56 -04:00
|
|
|
import Hidden from '@material-ui/core/Hidden';
|
2018-07-10 14:41:46 -04:00
|
|
|
import Typography from '@material-ui/core/Typography';
|
2018-06-11 13:18:59 -04:00
|
|
|
import ENSSubdomainRegistry from 'Embark/contracts/ENSSubdomainRegistry';
|
2018-07-09 17:53:56 -04:00
|
|
|
import { Button, Field, TextInput, MobileSearch, Card, Info, Text } from '../../ui/components'
|
2018-06-08 09:11:51 -04:00
|
|
|
import { IconCheck } from '../../ui/icons'
|
2018-06-06 15:37:03 -04:00
|
|
|
import theme from '../../ui/theme'
|
2018-06-05 16:48:30 -04:00
|
|
|
import { withFormik } from 'formik';
|
|
|
|
import PublicResolver from 'Embark/contracts/PublicResolver';
|
|
|
|
import { hash } from 'eth-ens-namehash';
|
2018-06-06 15:37:03 -04:00
|
|
|
import { CopyToClipboard } from 'react-copy-to-clipboard';
|
2018-06-11 13:18:59 -04:00
|
|
|
import RegisterSubDomain from '../ens/registerSubDomain';
|
2018-06-13 17:44:14 -04:00
|
|
|
import StatusLogo from '../../ui/icons/components/StatusLogo'
|
|
|
|
import EnsLogo from '../../ui/icons/logos/ens.png';
|
2018-07-13 15:08:21 -04:00
|
|
|
import { formatPrice } from '../ens/utils';
|
|
|
|
import CheckCircle from '../../ui/icons/components/baseline_check_circle_outline.png';
|
2018-06-19 10:12:49 -04:00
|
|
|
const { getPrice, getExpirationTime } = ENSSubdomainRegistry.methods;
|
2018-07-31 14:24:51 -04:00
|
|
|
import NotInterested from '@material-ui/icons/NotInterested';
|
2018-06-05 10:43:05 -04:00
|
|
|
|
2018-06-11 15:55:42 -04:00
|
|
|
const invalidSuffix = '0000000000000000000000000000000000000000'
|
2018-06-06 15:37:03 -04:00
|
|
|
const nullAddress = '0x0000000000000000000000000000000000000000'
|
2018-06-11 13:18:59 -04:00
|
|
|
const validAddress = address => address != nullAddress;
|
2018-06-11 15:55:42 -04:00
|
|
|
const validStatusAddress = address => !address.includes(invalidSuffix);
|
2018-06-05 16:48:30 -04:00
|
|
|
const formatName = domainName => domainName.includes('.') ? domainName : `${domainName}.stateofus.eth`;
|
2018-06-11 13:18:59 -04:00
|
|
|
const getDomain = fullDomain => formatName(fullDomain).split('.').slice(1).join('.');
|
|
|
|
const hashedDomain = domainName => hash(getDomain(domainName));
|
2018-06-20 11:13:33 -04:00
|
|
|
const { fromWei } = web3.utils;
|
2018-06-06 15:37:03 -04:00
|
|
|
|
2018-06-05 10:43:05 -04:00
|
|
|
const cardStyle = {
|
2018-06-12 18:17:04 -04:00
|
|
|
width: '100%',
|
2018-06-12 14:08:25 -04:00
|
|
|
padding: '30px',
|
|
|
|
height: '425px'
|
2018-06-05 10:43:05 -04:00
|
|
|
}
|
2018-06-04 20:34:56 -04:00
|
|
|
|
2018-06-06 15:37:03 -04:00
|
|
|
const addressStyle = {
|
|
|
|
fontSize: '18px',
|
|
|
|
fontWeight: 400,
|
2018-06-07 17:55:50 -04:00
|
|
|
cursor: 'copy',
|
2018-06-08 09:11:51 -04:00
|
|
|
wordWrap: 'break-word',
|
2018-06-06 15:37:03 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const backButton = {
|
|
|
|
fontSize: '40px',
|
2018-06-07 17:55:50 -04:00
|
|
|
color: theme.accent,
|
|
|
|
cursor: 'pointer'
|
2018-06-06 15:37:03 -04:00
|
|
|
}
|
|
|
|
|
2018-06-19 10:12:49 -04:00
|
|
|
const generatePrettyDate = (timestamp) => new Date(timestamp * 1000).toDateString();
|
|
|
|
|
2018-07-31 15:59:04 -04:00
|
|
|
const DisplayBox = ({ displayType, pubKey }) => (
|
|
|
|
<div style={{ border: '1px solid #EEF2F5', borderRadius: '8px', margin: '1em', display: 'flex', flexDirection: 'column', justifyContent: 'space-around', minHeight: '4em' }}>
|
|
|
|
<div style={{ margin: '3%', wordBreak: 'break-word' }}>
|
|
|
|
<div style={{ fontSize: '14px', color: '#939BA1' }}>{displayType}</div>
|
|
|
|
<Typography type='body1'>{pubKey}</Typography>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
2018-08-01 16:00:15 -04:00
|
|
|
const MobileAddressDisplay = ({ domainName, address, statusAccount, expirationTime, defaultAccount }) => (
|
|
|
|
<Fragment>
|
|
|
|
<Info background="#000000" style={{ margin: '0.4em', boxShadow: '0px 6px 10px rgba(0, 0, 0, 0.2)' }}>
|
|
|
|
<Typography variant="title" style={
|
|
|
|
{ display: 'flex', flexDirection: 'column', alignItems: 'center', justifyContent: 'space-evenly', height: '4em', color: '#ffffff', textAlign: 'center', margin: '10%' }
|
|
|
|
}>
|
|
|
|
<NotInterested style={{ marginBottom: '0.5em', fontSize: '2em' }}/>
|
|
|
|
<b>{formatName(domainName).toUpperCase()}</b>
|
|
|
|
<div style={{ fontWeight: 300 }}>
|
|
|
|
{expirationTime && <i>Expires {generatePrettyDate(expirationTime)}</i>}
|
|
|
|
</div>
|
|
|
|
</Typography>
|
|
|
|
</Info>
|
|
|
|
<Typography type='subheading' style={{ textAlign: 'center', fontSize: '26px', marginTop: '0.5em' }}>Name is unavailable</Typography>
|
|
|
|
<Typography type='body2' style={{ textAlign: 'center' }}>It is pointed to the following addresses</Typography>
|
|
|
|
<DisplayBox displayType='Wallet Address' pubKey={address} />
|
|
|
|
{validStatusAddress(statusAccount) && <DisplayBox displayType='Contact Code' pubKey={statusAccount} />}
|
|
|
|
</Fragment>
|
|
|
|
)
|
|
|
|
|
2018-06-12 11:11:28 -04:00
|
|
|
class RenderAddresses extends PureComponent {
|
|
|
|
state = { copied: false }
|
|
|
|
|
|
|
|
render() {
|
2018-08-01 16:00:15 -04:00
|
|
|
const { domainName, address, statusAccount, expirationTime, defaultAccount } = this.props
|
|
|
|
// TODO ADD CONDITIONAL RENDER TO MobileAddressDisplay
|
|
|
|
console.log(defaultAccount)
|
2018-06-12 11:11:28 -04:00
|
|
|
const { copied } = this.state
|
|
|
|
const markCopied = (v) => { this.setState({ copied: v }) }
|
|
|
|
const isCopied = address => address == copied;
|
|
|
|
const renderCopied = address => isCopied(address) && <span style={{ color: theme.positive }}><IconCheck/>Copied!</span>;
|
|
|
|
return (
|
2018-07-31 14:24:51 -04:00
|
|
|
<Fragment>
|
|
|
|
<Hidden mdDown>
|
|
|
|
<div style={{ display: 'flex', flexDirection: 'column' }}>
|
|
|
|
<Info.Action title="Click to copy"><b>{formatName(domainName).toUpperCase()}</b>{expirationTime && <i> (Expires {generatePrettyDate(expirationTime)})</i>} Resolves To:</Info.Action>
|
|
|
|
{address && <Text style={{ marginTop: '1em' }}>Ethereum Address {renderCopied(address)}</Text>}
|
|
|
|
<CopyToClipboard text={address} onCopy={markCopied}>
|
|
|
|
<div style={addressStyle}>{address}</div>
|
|
|
|
</CopyToClipboard>
|
|
|
|
{validStatusAddress(statusAccount) && <Text style={{ marginTop: '1em' }}>Status Address {renderCopied(statusAccount)}</Text>}
|
|
|
|
{validStatusAddress(statusAccount) && <CopyToClipboard text={statusAccount} onCopy={markCopied}>
|
|
|
|
<div style={{ ...addressStyle, color: isCopied ? theme.primary : null }}>{statusAccount}</div>
|
|
|
|
</CopyToClipboard>}
|
|
|
|
</div>
|
|
|
|
</Hidden>
|
|
|
|
<Hidden mdUp>
|
2018-08-01 16:00:15 -04:00
|
|
|
<MobileAddressDisplay {...this.props} />
|
2018-07-31 14:24:51 -04:00
|
|
|
</Hidden>
|
|
|
|
</Fragment>
|
2018-06-12 11:11:28 -04:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-10 14:41:46 -04:00
|
|
|
const RegisterInfoCard = ({ formattedDomain, domainPrice }) => (
|
|
|
|
<Fragment>
|
|
|
|
<Hidden mdDown>
|
|
|
|
<Info.Action title="No address is associated with this domain">
|
2018-07-10 15:23:21 -04:00
|
|
|
<span style={{ color: theme.accent }}>{formattedDomain.toLowerCase()}</span> can be registered for {!!domainPrice && formatPrice(fromWei(domainPrice))} SNT
|
2018-07-10 14:41:46 -04:00
|
|
|
</Info.Action>
|
|
|
|
</Hidden>
|
|
|
|
<Hidden mdUp>
|
2018-07-10 15:23:21 -04:00
|
|
|
<Info background="#415be3" style={{ margin: '0.4em' }}>
|
2018-07-10 14:41:46 -04:00
|
|
|
<Typography variant="title" style={
|
|
|
|
{ display: 'flex', flexDirection: 'column', alignItems: 'center', justifyContent: 'space-evenly', height: '4em', color: '#ffffff', textAlign: 'center', margin: '10%' }
|
|
|
|
}>
|
2018-07-13 15:08:21 -04:00
|
|
|
<img src={CheckCircle} style={{ maxWidth: '2.5em', marginBottom: '0.5em' }} />
|
2018-07-10 15:23:21 -04:00
|
|
|
<b>{formattedDomain.toLowerCase()}</b>
|
2018-07-13 15:08:21 -04:00
|
|
|
<div style={{ fontWeight: 300 }}>
|
|
|
|
{!!domainPrice && formatPrice(fromWei(domainPrice))} SNT / 1 year
|
2018-07-10 14:41:46 -04:00
|
|
|
</div>
|
|
|
|
</Typography>
|
|
|
|
</Info>
|
|
|
|
</Hidden>
|
2018-07-10 15:23:21 -04:00
|
|
|
<Hidden mdUp>
|
2018-07-11 14:44:32 -04:00
|
|
|
<Typography style={{ textAlign: 'center', padding: '1.5em' }}>
|
|
|
|
This name will be pointed to the wallet address and contact code below
|
2018-07-10 15:23:21 -04:00
|
|
|
</Typography>
|
|
|
|
</Hidden>
|
2018-07-10 14:41:46 -04:00
|
|
|
</Fragment>
|
|
|
|
)
|
|
|
|
|
2018-06-11 13:18:59 -04:00
|
|
|
class Register extends PureComponent {
|
|
|
|
state = { domainPrice: null };
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
const { domainName } = this.props;
|
|
|
|
getPrice(hashedDomain(domainName))
|
|
|
|
.call()
|
|
|
|
.then((res) => { this.setState({ domainPrice: res })});
|
|
|
|
}
|
|
|
|
|
2018-06-13 15:04:06 -04:00
|
|
|
onRegistered = (address, statusAccount) => {
|
|
|
|
const { domainPrice } = this.state;
|
|
|
|
const { subtractFromBalance } = this.props;
|
|
|
|
subtractFromBalance(domainPrice);
|
|
|
|
this.setState({ registered: { address, statusAccount } });
|
|
|
|
}
|
|
|
|
|
2018-06-11 13:18:59 -04:00
|
|
|
render() {
|
2018-07-13 15:29:11 -04:00
|
|
|
const { domainName, setStatus, style } = this.props;
|
2018-06-12 11:11:28 -04:00
|
|
|
const { domainPrice, registered } = this.state;
|
2018-06-11 15:55:42 -04:00
|
|
|
const formattedDomain = formatName(domainName);
|
|
|
|
const formattedDomainArray = formattedDomain.split('.')
|
2018-06-11 13:18:59 -04:00
|
|
|
return (
|
2018-07-13 15:29:11 -04:00
|
|
|
<div style={style}>
|
2018-06-12 11:11:28 -04:00
|
|
|
{!registered ?
|
|
|
|
<Fragment>
|
2018-07-10 14:41:46 -04:00
|
|
|
<RegisterInfoCard {...{ formattedDomain, domainPrice }}/>
|
2018-06-12 11:11:28 -04:00
|
|
|
<RegisterSubDomain
|
|
|
|
subDomain={formattedDomainArray[0]}
|
|
|
|
domainName={formattedDomainArray.slice(1).join('.')}
|
|
|
|
domainPrice={domainPrice}
|
2018-06-13 15:04:06 -04:00
|
|
|
registeredCallbackFn={this.onRegistered} />
|
2018-06-12 11:11:28 -04:00
|
|
|
</Fragment> :
|
|
|
|
<RenderAddresses {...this.props} address={registered.address} statusAccount={registered.statusAccount} />}
|
2018-06-11 13:18:59 -04:00
|
|
|
<div style={backButton} onClick={() => setStatus(null)}>←</div>
|
2018-07-13 15:29:11 -04:00
|
|
|
</div>
|
2018-06-11 13:18:59 -04:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-13 15:04:06 -04:00
|
|
|
const mapDispatchToProps = dispatch => ({
|
|
|
|
subtractFromBalance(amount) {
|
|
|
|
dispatch(accountActions.subtractfromSntTokenBalance(amount));
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2018-08-01 16:00:15 -04:00
|
|
|
const mapStateToProps = state => ({
|
|
|
|
defaultAccount: getDefaultAccount(state)
|
|
|
|
})
|
|
|
|
|
|
|
|
const ConnectedRegister = connect(mapStateToProps, mapDispatchToProps)(Register);
|
2018-06-13 15:04:06 -04:00
|
|
|
|
2018-08-01 16:00:15 -04:00
|
|
|
const DisplayAddress = connect(mapStateToProps)((props) => (
|
2018-06-12 11:11:28 -04:00
|
|
|
<Fragment>
|
2018-06-12 14:08:25 -04:00
|
|
|
{validAddress(props.address) ?
|
|
|
|
<RenderAddresses {...props} />
|
2018-06-12 11:11:28 -04:00
|
|
|
:
|
2018-07-10 14:41:46 -04:00
|
|
|
<Hidden mdUp>
|
|
|
|
<Info.Action title="No address is associated with this domain">
|
|
|
|
{props.domainName.toUpperCase()}
|
|
|
|
</Info.Action>
|
|
|
|
</Hidden>
|
|
|
|
}
|
2018-06-12 14:08:25 -04:00
|
|
|
<div style={backButton} onClick={() => props.setStatus(null)}>←</div>
|
2018-06-12 11:11:28 -04:00
|
|
|
</Fragment>
|
2018-08-01 16:00:15 -04:00
|
|
|
))
|
2018-06-06 15:37:03 -04:00
|
|
|
|
2018-07-13 15:29:11 -04:00
|
|
|
const LookupForm = ({ handleSubmit, values, handleChange, justSearch }) => (
|
2018-07-09 17:53:56 -04:00
|
|
|
<Fragment>
|
2018-07-13 13:59:18 -04:00
|
|
|
<form onSubmit={handleSubmit}>
|
2018-07-10 14:41:46 -04:00
|
|
|
<Hidden mdDown>
|
2018-07-10 10:15:26 -04:00
|
|
|
<Field label="Enter Domain or Status Name" wide>
|
|
|
|
<TextInput
|
|
|
|
value={values.domainName}
|
|
|
|
name="domainName"
|
|
|
|
onChange={handleChange}
|
|
|
|
wide
|
|
|
|
required />
|
|
|
|
</Field>
|
|
|
|
</Hidden>
|
2018-07-10 14:41:46 -04:00
|
|
|
<Hidden mdUp>
|
2018-07-13 13:59:18 -04:00
|
|
|
<MobileSearch
|
|
|
|
search
|
|
|
|
name="domainName"
|
|
|
|
placeholder='Search for vacant name'
|
|
|
|
value={values.domainName}
|
|
|
|
onChange={handleChange}
|
|
|
|
required
|
|
|
|
wide />
|
2018-07-13 15:29:11 -04:00
|
|
|
{!justSearch && <Typography variant="subheading" style={{ color: '#939ba1', textAlign: 'center', marginTop: '25vh' }}>
|
2018-07-13 13:59:18 -04:00
|
|
|
Symbols * / <br/>
|
|
|
|
are not supported
|
2018-07-13 15:29:11 -04:00
|
|
|
</Typography>}
|
2018-07-10 10:15:26 -04:00
|
|
|
</Hidden>
|
2018-07-10 14:41:46 -04:00
|
|
|
<Hidden mdDown>
|
2018-07-10 10:15:26 -04:00
|
|
|
<Button mode="strong" type="submit" wide>
|
|
|
|
Lookup Address
|
|
|
|
</Button>
|
|
|
|
</Hidden>
|
2018-07-09 17:53:56 -04:00
|
|
|
</form>
|
|
|
|
</Fragment>
|
|
|
|
)
|
|
|
|
|
2018-06-05 16:48:30 -04:00
|
|
|
const InnerForm = ({
|
|
|
|
values,
|
|
|
|
errors,
|
|
|
|
touched,
|
|
|
|
handleChange,
|
|
|
|
handleBlur,
|
|
|
|
handleSubmit,
|
|
|
|
isSubmitting,
|
2018-06-06 15:37:03 -04:00
|
|
|
status,
|
|
|
|
setStatus
|
2018-06-05 16:48:30 -04:00
|
|
|
}) => (
|
2018-07-13 15:29:11 -04:00
|
|
|
<div>
|
2018-07-10 14:41:46 -04:00
|
|
|
<Hidden mdDown>
|
2018-07-10 10:15:26 -04:00
|
|
|
<span style={{ display: 'flex', justifyContent: 'space-evenly', marginBottom: '10px' }}>
|
|
|
|
<StatusLogo />
|
|
|
|
<img style={{ maxWidth: '150px', alignSelf: 'center' }} src={EnsLogo} alt="Ens Logo"/>
|
|
|
|
</span>
|
2018-07-09 17:53:56 -04:00
|
|
|
</Hidden>
|
2018-06-11 13:18:59 -04:00
|
|
|
{!status
|
2018-07-09 17:53:56 -04:00
|
|
|
? <LookupForm {...{ handleSubmit, values, handleChange }} />
|
2018-06-11 13:18:59 -04:00
|
|
|
: validAddress(status.address) ?
|
|
|
|
<DisplayAddress
|
|
|
|
domainName={values.domainName}
|
|
|
|
address={status.address}
|
|
|
|
statusAccount={status.statusAccount}
|
2018-06-19 10:12:49 -04:00
|
|
|
expirationTime={status.expirationTime}
|
2018-06-11 13:18:59 -04:00
|
|
|
setStatus={setStatus} /> :
|
2018-07-13 15:29:11 -04:00
|
|
|
<div>
|
|
|
|
<LookupForm {...{ handleSubmit, values, handleChange }} justSearch />
|
|
|
|
<ConnectedRegister
|
2018-07-30 21:11:20 -04:00
|
|
|
style={{ position: 'relative' }}
|
2018-07-13 15:29:11 -04:00
|
|
|
setStatus={setStatus}
|
|
|
|
domainName={values.domainName} />
|
|
|
|
</div>
|
2018-06-11 13:18:59 -04:00
|
|
|
}
|
2018-07-09 17:53:56 -04:00
|
|
|
</div>
|
2018-06-04 20:34:56 -04:00
|
|
|
)
|
|
|
|
|
2018-06-05 16:48:30 -04:00
|
|
|
const NameLookup = withFormik({
|
|
|
|
mapPropsToValues: props => ({ domainName: '' }),
|
2018-06-07 17:55:50 -04:00
|
|
|
async handleSubmit(values, { status, setSubmitting, setStatus }) {
|
2018-06-05 16:48:30 -04:00
|
|
|
const { domainName } = values;
|
2018-06-07 17:55:50 -04:00
|
|
|
const { addr, text } = PublicResolver.methods;
|
|
|
|
const lookupHash = hash(formatName(domainName));
|
|
|
|
const address = await addr(lookupHash).call();
|
|
|
|
const statusAccount = await text(lookupHash, 'statusAccount').call();
|
2018-06-19 10:12:49 -04:00
|
|
|
const expirationTime = await getExpirationTime(lookupHash).call();
|
|
|
|
setStatus({ address, statusAccount, expirationTime });
|
2018-06-05 16:48:30 -04:00
|
|
|
}
|
|
|
|
})(InnerForm)
|
|
|
|
|
2018-06-04 20:34:56 -04:00
|
|
|
export default NameLookup;
|