const Component = require('react').Component const h = require('react-hyperscript') const inherits = require('util').inherits const EtherBalance = require('./eth-balance') const addressSummary = require('../util').addressSummary const explorerLink = require('../../lib/explorer-link') const CopyButton = require('./copyButton') const vreme = new (require('vreme')) const extension = require('../../../app/scripts/lib/extension') const Tooltip = require('./tooltip') const TransactionIcon = require('./transaction-list-item-icon') const ShiftListItem = require('./shift-list-item') module.exports = TransactionListItem inherits(TransactionListItem, Component) function TransactionListItem () { Component.call(this) } TransactionListItem.prototype.render = function () { const { transaction, network } = this.props if (transaction.key === 'shapeshift') { if (network === '1') return h(ShiftListItem, transaction) } var date = formatDate(transaction.time) let isLinkable = false const numericNet = parseInt(network) isLinkable = numericNet === 1 || numericNet === 3 var isMsg = ('msgParams' in transaction) var isTx = ('txParams' in transaction) var isPending = transaction.status === 'unapproved' let txParams if (isTx) { txParams = transaction.txParams } else if (isMsg) { txParams = transaction.msgParams } const isClickable = ('hash' in transaction && isLinkable) || isPending return ( h(`.transaction-list-item.flex-row.flex-space-between${isClickable ? '.pointer' : ''}`, { onClick: (event) => { if (isPending) { this.props.showTx(transaction.id) } event.stopPropagation() if (!transaction.hash || !isLinkable) return var url = explorerLink(transaction.hash, parseInt(network)) extension.tabs.create({ url }) }, style: { padding: '20px 0', }, }, [ h('.identicon-wrapper.flex-column.flex-center.select-none', [ h('.pop-hover', { onClick: (event) => { event.stopPropagation() if (!isTx || isPending) return var url = `https://metamask.github.io/eth-tx-viz/?tx=${transaction.hash}` extension.tabs.create({ url }) }, }, [ h(TransactionIcon, { txParams, transaction, isTx, isMsg }), ]), ]), h('.flex-column', {style: {width: '200px', overflow: 'hidden'}}, [ domainField(txParams), h('div', date), recipientField(txParams, transaction, isTx, isMsg), ]), // Places a copy button if tx is successful, else places a placeholder empty div. transaction.hash ? h(CopyButton, { value: transaction.hash }) : h('div', {style: { display: 'flex', alignItems: 'center', width: '26px' }}), isTx ? h(EtherBalance, { value: txParams.value, width: '55px', shorten: true, showFiat: false, style: {fontSize: '15px'}, }) : h('.flex-column'), ]) ) } function domainField (txParams) { return h('div', { style: { fontSize: 'x-small', color: '#ABA9AA', overflow: 'hidden', textOverflow: 'ellipsis', width: '100%', }, }, [ txParams.origin, ]) } function recipientField (txParams, transaction, isTx, isMsg) { let message if (isMsg) { message = 'Signature Requested' } else if (txParams.to) { message = addressSummary(txParams.to) } else { message = 'Contract Published' } return h('div', { style: { fontSize: 'x-small', color: '#ABA9AA', }, }, [ message, failIfFailed(transaction), ]) } function formatDate (date) { return vreme.format(new Date(date), 'March 16 2014 14:30') } function failIfFailed (transaction) { if (transaction.status === 'rejected') { return h('span.error', ' (Rejected)') } if (transaction.err) { return h(Tooltip, { title: transaction.err.message, position: 'bottom', }, [ h('span.error', ' (Failed)'), ]) } }