aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDan Finlay <somniac@me.com>2016-05-27 06:19:07 +0800
committerDan Finlay <somniac@me.com>2016-05-27 06:19:07 +0800
commitf1fb7ff42c5a86411108fe908d5d85921dcdaec4 (patch)
tree3ef4c7eefadc296203ba34f653d3a19aced1a255
parent7f4929a2d28b382c64761bd456a19e5e43ac4f64 (diff)
parent01e5bc25a9e03ab2d9c72dad29b5d7fa03933009 (diff)
downloadtangerine-wallet-browser-f1fb7ff42c5a86411108fe908d5d85921dcdaec4.tar
tangerine-wallet-browser-f1fb7ff42c5a86411108fe908d5d85921dcdaec4.tar.gz
tangerine-wallet-browser-f1fb7ff42c5a86411108fe908d5d85921dcdaec4.tar.bz2
tangerine-wallet-browser-f1fb7ff42c5a86411108fe908d5d85921dcdaec4.tar.lz
tangerine-wallet-browser-f1fb7ff42c5a86411108fe908d5d85921dcdaec4.tar.xz
tangerine-wallet-browser-f1fb7ff42c5a86411108fe908d5d85921dcdaec4.tar.zst
tangerine-wallet-browser-f1fb7ff42c5a86411108fe908d5d85921dcdaec4.zip
Merge pull request #224 from MetaMask/PendingTxList
Add pending txs to tx list
-rw-r--r--CHANGELOG.md3
-rw-r--r--app/manifest.json2
-rw-r--r--ui/app/account-detail.js3
-rw-r--r--ui/app/actions.js9
-rw-r--r--ui/app/components/transaction-list-item-icon.js46
-rw-r--r--ui/app/components/transaction-list-item.js117
-rw-r--r--ui/app/components/transaction-list.js8
-rw-r--r--ui/app/css/index.css2
-rw-r--r--ui/app/css/lib.css19
-rw-r--r--ui/app/reducers/app.js28
10 files changed, 175 insertions, 62 deletions
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 3b0469b19..e59f7ded3 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -2,6 +2,9 @@
## Current Master
+- Added pending transactions to transaction list on account screen.
+- Clicking a pending transaction takes you back to the transaction approval screen.
+
## 2.1.0 2016-05-26
- Added copy address button to account list.
diff --git a/app/manifest.json b/app/manifest.json
index c3417103e..5b1be504d 100644
--- a/app/manifest.json
+++ b/app/manifest.json
@@ -1,7 +1,7 @@
{
"name": "__MSG_appName__",
"short_name": "Metamask",
- "version": "2.0.1",
+ "version": "2.1.0",
"manifest_version": 2,
"description": "__MSG_appDescription__",
"icons": {
diff --git a/ui/app/account-detail.js b/ui/app/account-detail.js
index 1dcce1d08..2f412c5be 100644
--- a/ui/app/account-detail.js
+++ b/ui/app/account-detail.js
@@ -201,6 +201,9 @@ AccountDetailScreen.prototype.transactionList = function() {
network,
unconfTxs,
unconfMsgs,
+ viewPendingTx:(txId) => {
+ this.props.dispatch(actions.viewPendingTx(txId))
+ }
})
}
diff --git a/ui/app/actions.js b/ui/app/actions.js
index 5b058aaed..ae6125b20 100644
--- a/ui/app/actions.js
+++ b/ui/app/actions.js
@@ -76,6 +76,8 @@ var actions = {
txError: txError,
nextTx: nextTx,
previousTx: previousTx,
+ viewPendingTx: viewPendingTx,
+ VIEW_PENDING_TX: 'VIEW_PENDING_TX',
// app messages
showAccountDetail: showAccountDetail,
BACK_TO_ACCOUNT_DETAIL: 'BACK_TO_ACCOUNT_DETAIL',
@@ -387,6 +389,13 @@ function nextTx() {
}
}
+function viewPendingTx(txId) {
+ return {
+ type: actions.VIEW_PENDING_TX,
+ value: txId,
+ }
+}
+
function previousTx() {
return {
type: actions.PREVIOUS_TX,
diff --git a/ui/app/components/transaction-list-item-icon.js b/ui/app/components/transaction-list-item-icon.js
new file mode 100644
index 000000000..fbee4b218
--- /dev/null
+++ b/ui/app/components/transaction-list-item-icon.js
@@ -0,0 +1,46 @@
+const Component = require('react').Component
+const h = require('react-hyperscript')
+const inherits = require('util').inherits
+
+const Identicon = require('./identicon')
+
+module.exports = TransactionIcon
+
+
+inherits(TransactionIcon, Component)
+function TransactionIcon() {
+ Component.call(this)
+}
+
+TransactionIcon.prototype.render = function() {
+ const { transaction, txParams, isTx, isMsg } = this.props
+
+ if (transaction.status === 'rejected') {
+ return h('i.fa.fa-exclamation-triangle.fa-lg.error', {
+ style: {
+ width: '24px',
+ }
+ })
+ }
+
+ if (isMsg) {
+ return h('i.fa.fa-certificate.fa-lg', {
+ style: {
+ width: '24px',
+ }
+ })
+ }
+
+ if (txParams.to) {
+ return h(Identicon, {
+ diameter: 24,
+ address: txParams.to || transaction.hash,
+ })
+ } else {
+ return h('i.fa.fa-file-text-o.fa-lg', {
+ style: {
+ width: '24px',
+ }
+ })
+ }
+}
diff --git a/ui/app/components/transaction-list-item.js b/ui/app/components/transaction-list-item.js
index a0715db0b..fc1c3c630 100644
--- a/ui/app/components/transaction-list-item.js
+++ b/ui/app/components/transaction-list-item.js
@@ -2,13 +2,14 @@ const Component = require('react').Component
const h = require('react-hyperscript')
const inherits = require('util').inherits
-const Identicon = require('./identicon')
const EtherBalance = require('./eth-balance')
const addressSummary = require('../util').addressSummary
const explorerLink = require('../../lib/explorer-link')
const formatBalance = require('../util').formatBalance
const vreme = new (require('vreme'))
+const TransactionIcon = require('./transaction-list-item-icon')
+
module.exports = TransactionListItem
@@ -18,20 +19,36 @@ function TransactionListItem() {
}
TransactionListItem.prototype.render = function() {
- const { transaction, i } = this.props
+ const { transaction, i, network } = this.props
var date = formatDate(transaction.time)
+ let isLinkable = false
+ const numericNet = parseInt(network)
+ isLinkable = numericNet === 1 || numericNet === 2
+
var isMsg = ('msgParams' in transaction)
var isTx = ('txParams' in transaction)
+ var isPending = transaction.status === 'unconfirmed'
+
+ let txParams
+ if (isTx) {
+ txParams = transaction.txParams
+ } else if (isMsg) {
+ txParams = transaction.msgParams
+ }
- var txParams = transaction.txParams
+ const isClickable = ('hash' in transaction && isLinkable) || isPending
return (
- h(`.transaction-list-item.flex-row.flex-space-between${transaction.hash ? '.pointer' : ''}`, {
+ h(`.transaction-list-item.flex-row.flex-space-between${isClickable ? '.pointer' : ''}`, {
key: `tx-${transaction.id + i}`,
onClick: (event) => {
- if (!transaction.hash) return
+ if (isPending) {
+ this.props.showTx(transaction.id)
+ }
+
+ if (!transaction.hash || !isLinkable) return
var url = explorerLink(transaction.hash, parseInt(network))
chrome.tabs.create({ url })
},
@@ -42,53 +59,59 @@ TransactionListItem.prototype.render = function() {
// large identicon
h('.identicon-wrapper.flex-column.flex-center.select-none', [
- identicon(txParams, transaction),
+ transaction.status === 'unconfirmed' ? h('.red-dot', ' ') :
+ h(TransactionIcon, { txParams, transaction, isTx, isMsg }),
]),
h('.flex-column', [
-
+ domainField(txParams),
h('div', date),
-
- recipientField(txParams, transaction),
-
+ recipientField(txParams, transaction, isTx, isMsg),
]),
- h(EtherBalance, {
+ isTx ? h(EtherBalance, {
value: txParams.value,
- }),
+ }) : h('.flex-column'),
])
)
}
+function domainField(txParams) {
+ return h('div', {
+ style: {
+ fontSize: 'small',
+ color: '#ABA9AA',
+ },
+ },[
+ txParams.origin,
+ ])
+}
-function recipientField(txParams, transaction) {
- if (txParams.to) {
- return h('div', {
- style: {
- fontSize: 'small',
- color: '#ABA9AA',
- },
- }, [
- addressSummary(txParams.to),
- failIfFailed(transaction),
- ])
+function recipientField(txParams, transaction, isTx, isMsg) {
+ let message
+ if (isMsg) {
+ message = 'Signature Requested'
+ } else if (txParams.to) {
+ message = addressSummary(txParams.to)
} else {
-
- return h('div', {
- style: {
- fontSize: 'small',
- color: '#ABA9AA',
- },
- },[
- 'Contract Published',
- failIfFailed(transaction),
- ])
+ message = 'Contract Published'
}
+
+ return h('div', {
+ style: {
+ fontSize: 'small',
+ color: '#ABA9AA',
+ },
+ },[
+ message,
+ failIfFailed(transaction),
+ ])
+
}
TransactionListItem.prototype.renderMessage = function() {
- const { transaction, i } = this.props
+ const { transaction, i, network } = this.props
return h('div', 'wowie, thats a message')
}
@@ -96,31 +119,11 @@ function formatDate(date){
return vreme.format(new Date(date), 'March 16 2014 14:30')
}
-function identicon(txParams, transaction) {
- if (transaction.status === 'rejected') {
- return h('i.fa.fa-exclamation-triangle.fa-lg.error', {
- style: {
- width: '24px',
- }
- })
- }
-
- if (txParams.to) {
- return h(Identicon, {
- diameter: 24,
- address: txParams.to || transaction.hash,
- })
- } else {
- return h('i.fa.fa-file-text-o.fa-lg', {
- style: {
- width: '24px',
- }
- })
- }
-}
-
function failIfFailed(transaction) {
if (transaction.status === 'rejected') {
+ return h('span.error', ' (Rejected)')
+ }
+ if (transaction.status === 'failed') {
return h('span.error', ' (Failed)')
}
}
diff --git a/ui/app/components/transaction-list.js b/ui/app/components/transaction-list.js
index 86abd9709..ed2e1ee0a 100644
--- a/ui/app/components/transaction-list.js
+++ b/ui/app/components/transaction-list.js
@@ -14,7 +14,8 @@ function TransactionList() {
TransactionList.prototype.render = function() {
const { txsToRender, network, unconfTxs, unconfMsgs } = this.props
- const transactions = txsToRender
+ const transactions = txsToRender.concat(unconfMsgs)
+ .sort((a, b) => b.time - a.time)
return (
@@ -51,7 +52,10 @@ TransactionList.prototype.render = function() {
transactions.length ?
transactions.map((transaction, i) => {
return h(TransactionListItem, {
- transaction, i
+ transaction, i, network,
+ showTx:(txId) => {
+ this.props.viewPendingTx(txId)
+ },
})
})
:
diff --git a/ui/app/css/index.css b/ui/app/css/index.css
index 8e25c1f88..650c52cef 100644
--- a/ui/app/css/index.css
+++ b/ui/app/css/index.css
@@ -296,6 +296,8 @@ input.large-input {
.identity-panel .identicon-wrapper {
margin: 4px;
margin-top: 8px;
+ display: flex;
+ align-items: center;
}
.identity-panel .identicon-wrapper span {
diff --git a/ui/app/css/lib.css b/ui/app/css/lib.css
index 73be4023e..865d8060c 100644
--- a/ui/app/css/lib.css
+++ b/ui/app/css/lib.css
@@ -167,6 +167,20 @@ hr.horizontal-line {
background: white;
}
+.red-dot {
+ position: inherit;
+ background: red;
+ color: white;
+ border-radius: 10px;
+ height: 12px;
+ min-width: 12px;
+ margin-left: 6px;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ padding: 4px;
+}
+
.pending-dot {
background: red;
left: 57px;
@@ -180,3 +194,8 @@ hr.horizontal-line {
justify-content: center;
padding: 4px;
}
+
+.ether-balance {
+ display: flex;
+ align-items: center;
+}
diff --git a/ui/app/reducers/app.js b/ui/app/reducers/app.js
index a98b809d6..08c2268c1 100644
--- a/ui/app/reducers/app.js
+++ b/ui/app/reducers/app.js
@@ -12,10 +12,10 @@ function reduceApp(state, action) {
const pendingTxs = hasPendingTxs(state)
let name = 'accounts'
if (selectedAccount) {
- defaultView = 'accountDetail'
+ name = 'accountDetail'
}
if (pendingTxs) {
- defaultView = 'confTx'
+ name = 'confTx'
}
var defaultView = {
@@ -270,6 +270,17 @@ function reduceApp(state, action) {
}
})
+ case actions.VIEW_PENDING_TX:
+ const context = indexForPending(state, action.value)
+ return extend(appState, {
+ transForward: true,
+ currentView: {
+ name: 'confTx',
+ context,
+ warning: null,
+ }
+ })
+
case actions.PREVIOUS_TX:
return extend(appState, {
transForward: false,
@@ -366,3 +377,16 @@ function hasPendingTxs (state) {
var unconfTxList = txHelper(unconfTxs, unconfMsgs)
return unconfTxList.length > 0
}
+
+function indexForPending(state, txId) {
+ var unconfTxs = state.metamask.unconfTxs
+ var unconfMsgs = state.metamask.unconfMsgs
+ var unconfTxList = txHelper(unconfTxs, unconfMsgs)
+ let idx
+ unconfTxList.forEach((tx, i) => {
+ if (tx.id === txId) {
+ idx = i
+ }
+ })
+ return idx
+}