openethereum/js/packages/ui/AccountCard/accountCard.js
Jaco Greeff 49fdd23d58 Ui 2 move to packages/* (#6113)
* Move secureApi to shell

* Extract isTestnet test

* Use mobx + subscriptions for status

* Re-add status indicator

* Add lerna

* Move intial packages to js/packages

* Move 3rdparty/{email,sms}-verification to correct location

* Move package.json & README to library src

* Move tests for library packages

* Move views & dapps to packages

* Move i18n to root

* Move shell to actual src (main app)

* Remove ~ references

* Change ~ to root (explicit imports)

* Finalise convert of ~

* Move views into dapps as well

* Move dapps to packages/

* Fix references

* Update css

* Update test spec locations

* Update tests

* Case fix

* Skip flakey tests

* Update enzyme

* Skip previously ignored tests

* Allow empty api for hw

* Re-add theme for embed
2017-07-21 15:46:53 +02:00

221 lines
5.6 KiB
JavaScript

// Copyright 2015-2017 Parity Technologies (UK) Ltd.
// This file is part of Parity.
// Parity is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// Parity is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
import React, { Component } from 'react';
import PropTypes from 'prop-types';
import ReactDOM from 'react-dom';
import keycode from 'keycode';
import Balance from '../Balance';
import Container from '../Container';
import IdentityIcon from '../IdentityIcon';
import IdentityName from '../IdentityName';
import Tags from '../Tags';
import styles from './accountCard.css';
export default class AccountCard extends Component {
static propTypes = {
children: PropTypes.node,
account: PropTypes.object.isRequired,
balance: PropTypes.object,
className: PropTypes.string,
disableAddressClick: PropTypes.bool,
onClick: PropTypes.func,
onFocus: PropTypes.func
};
static defaultProps = {
disableAddressClick: false
};
state = {
copied: false
};
render () {
const { account, balance, className, onFocus, children } = this.props;
const { copied } = this.state;
const { address, description, meta = {}, name } = account;
const { tags = [] } = meta;
const classes = [ styles.account, className ];
if (copied) {
classes.push(styles.copied);
}
const props = onFocus
? { tabIndex: 0 }
: {};
return (
<Container
key={ address }
className={ classes.join(' ') }
onClick={ this.onClick }
onFocus={ this.onFocus }
onKeyDown={ this.handleKeyDown }
{ ...props }
>
<div className={ styles.mainContainer }>
<div className={ styles.infoContainer }>
<IdentityIcon address={ address } />
<div className={ styles.accountInfo }>
<div className={ styles.accountName }>
<IdentityName
address={ address }
name={ name }
unknown
/>
</div>
{ this.renderDescription(description) }
{ this.renderAddress(address) }
</div>
</div>
<Balance
address={ address }
balance={ balance }
className={ styles.balance }
showOnlyEth
/>
{ children }
</div>
{
tags && tags.length > 0
? (
<div className={ styles.tagsContainer }>
<div className={ styles.tags }>
<Tags
floating={ false }
horizontal
tags={ tags }
/>
</div>
</div>
) : null
}
</Container>
);
}
renderDescription (description) {
if (!description) {
return null;
}
return (
<div className={ styles.description }>
<span>{ description }</span>
</div>
);
}
renderAddress (address) {
return (
<div className={ styles.addressContainer }>
<span
className={ styles.address }
onClick={ this.handleAddressClick }
ref={ `address` }
title={ address }
>
{ address }
</span>
</div>
);
}
handleAddressClick = (event) => {
const { disableAddressClick } = this.props;
// Stop the event if address click is disallowed
if (disableAddressClick) {
return this.preventEvent(event);
}
return this.onClick(event);
}
handleKeyDown = (event) => {
const codeName = keycode(event);
if (event.ctrlKey) {
// Copy the selected address if nothing selected and there is
// a focused item
const isSelection = !window.getSelection || window.getSelection().type === 'Range';
if (codeName === 'c' && !isSelection) {
const element = ReactDOM.findDOMNode(this.refs.address);
// Copy the address from the right element
// @see https://developers.google.com/web/updates/2015/04/cut-and-copy-commands
try {
const range = document.createRange();
range.selectNode(element);
window.getSelection().addRange(range);
document.execCommand('copy');
try {
window.getSelection().removeRange(range);
} catch (e) {
window.getSelection().removeAllRanges();
}
this.setState({ copied: true }, () => {
window.setTimeout(() => {
this.setState({ copied: false });
}, 250);
});
} catch (e) {
console.warn('could not copy');
}
}
return event;
}
}
onClick = (event) => {
const { account, onClick } = this.props;
// Stop the default event if text is selected
if (window.getSelection && window.getSelection().type === 'Range') {
return this.preventEvent(event);
}
onClick && onClick(account.address);
}
onFocus = () => {
const { account, onFocus } = this.props;
onFocus && onFocus(account.index);
}
preventEvent = (event) => {
event.preventDefault();
event.stopPropagation();
}
setTagRef = (tagRef) => {
this.tagRefs.push(tagRef);
}
}