// 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 . import React, { Component, PropTypes } from 'react'; import ReactDOM from 'react-dom'; import keycode from 'keycode'; import Balance from '~/ui/Balance'; import IdentityIcon from '~/ui/IdentityIcon'; import IdentityName from '~/ui/IdentityName'; import Tags from '~/ui/Tags'; import styles from './accountCard.css'; export default class AccountCard extends Component { static propTypes = { account: PropTypes.object.isRequired, allowAddressClick: PropTypes.bool, balance: PropTypes.object, className: PropTypes.string, onClick: PropTypes.func, onFocus: PropTypes.func }; static defaultProps = { allowAddressClick: false }; state = { copied: false }; render () { const { account, balance, className, onFocus } = 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 (
{ this.renderDescription(description) } { this.renderAddress(address) }
{ tags && tags.length > 0 ? (
) : null }
); } renderDescription (description) { if (!description) { return null; } return (
{ description }
); } renderAddress (address) { return (
{ address }
); } handleAddressClick = (event) => { const { allowAddressClick } = this.props; // Don't stop the event if address click is allowed if (allowAddressClick) { return this.onClick(event); } return this.preventEvent(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); } }