* s/Delete Contract/Forget Contract/ (#4237) * Adjust the location of the signer snippet (#4155) * Additional building-block UI components (#4239) * Currency WIP * Expand tests * Pass className * Add QrCode * Export new components in ~/ui * s/this.props.netSymbol/netSymbol/ * Fix import case * ui/SectionList component (#4292) * array chunking utility * add SectionList component * Add TODOs to indicate possible future work * Add missing overlay style (as used in dapps at present) * Add a Playground for the UI Components (#4301) * Playground // WIP * Linting * Add Examples with code * CSS Linting * Linting * Add Connected Currency Symbol * 2015-2017 * 2015-2017 * 2015-2017 * 2015-2017 * 2015-2017 * 2015-2017 * 2015-2017 * Added `renderSymbol` tests * PR grumbles * Add Eth and Btc QRCode examples * 2015-2017 * Add tests for playground * Fixing tests * Split Dapp icon into ui/DappIcon (#4308) * Add QrCode & Copy to ShapeShift (#4322) * Extract CopyIcon to ~/ui/Icons * Add copy & QrCode address * Default size 4 * Add bitcoin: link * use protocol links applicable to coin exchanged * Remove .only * Display QrCode for accounts, addresses & contracts (#4329) * Allow Portal to be used as top-level modal (#4338) * Portal * Allow Portal to be used in as both top-level and popover * modal/popover variable naming * export Portal in ~/ui * Properly handle optional onKeyDown * Add simple Playground Example * Add proper event listener to Portal (#4359) * Display AccountCard name via IdentityName (#4235) * Fix signing (#4363) * Dapp Account Selection & Defaults (#4355) * Add parity_defaultAccount RPC (with subscription) (#4383) * Default Account selector in Signer overlay (#4375) * Typo, fixes #4271 (#4391) * Fix ParityBar account selection overflows (#4405) * Available Dapp selection alignment with Permissions (Portal) (#4374) * registry dapp: make lookup use lower case (#4409) * Dapps use defaultAccount instead of own selectors (#4386) * Poll for defaultAccount to update dapp & overlay subscriptions (#4417) * Poll for defaultAccount (Fixes #4413) * Fix nextTimeout on catch * Store timers * Re-enable default updates on change detection * Add block & timestamp conditions to Signer (#4411) * Extension installation overlay (#4423) * Extension installation overlay * Pr gumbles * Spelling * Update Chrome URL * Fix for non-included jsonrpc * Extend Portal component (as per Modal) #4392
175 lines
4.5 KiB
JavaScript
175 lines
4.5 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, 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,
|
|
balance: PropTypes.object,
|
|
className: PropTypes.string,
|
|
onClick: PropTypes.func,
|
|
onFocus: PropTypes.func
|
|
};
|
|
|
|
state = {
|
|
copied: false
|
|
};
|
|
|
|
render () {
|
|
const { account, balance, className } = 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);
|
|
}
|
|
|
|
return (
|
|
<div
|
|
key={ address }
|
|
tabIndex={ 0 }
|
|
className={ classes.join(' ') }
|
|
onClick={ this.onClick }
|
|
onFocus={ this.onFocus }
|
|
onKeyDown={ this.handleKeyDown }
|
|
>
|
|
<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>
|
|
|
|
<Tags tags={ tags } />
|
|
<Balance
|
|
balance={ balance }
|
|
className={ styles.balance }
|
|
showOnlyEth
|
|
showZeroValues
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
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.preventEvent }
|
|
ref={ `address` }
|
|
title={ address }
|
|
>
|
|
{ address }
|
|
</span>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
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 = () => {
|
|
const { account, onClick } = this.props;
|
|
|
|
onClick && onClick(account.address);
|
|
}
|
|
|
|
onFocus = () => {
|
|
const { account, onFocus } = this.props;
|
|
|
|
onFocus && onFocus(account.index);
|
|
}
|
|
|
|
preventEvent = (e) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
}
|
|
|
|
setTagRef = (tagRef) => {
|
|
this.tagRefs.push(tagRef);
|
|
}
|
|
}
|