1 // Copyright (c) 2011-2016 The Bitcoin Core developers
2 // Distributed under the MIT software license, see the accompanying
3 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
5 #include "transactiondesc.h"
7 #include "bitcoinunits.h"
9 #include "paymentserver.h"
10 #include "transactionrecord.h"
13 #include "consensus/consensus.h"
14 #include "validation.h"
15 #include "script/script.h"
18 #include "wallet/db.h"
19 #include "wallet/wallet.h"
24 QString
TransactionDesc::FormatTxStatus(const CWalletTx
& wtx
)
26 AssertLockHeld(cs_main
);
27 if (!CheckFinalTx(wtx
))
29 if (wtx
.tx
->nLockTime
< LOCKTIME_THRESHOLD
)
30 return tr("Open for %n more block(s)", "", wtx
.tx
->nLockTime
- chainActive
.Height());
32 return tr("Open until %1").arg(GUIUtil::dateTimeStr(wtx
.tx
->nLockTime
));
36 int nDepth
= wtx
.GetDepthInMainChain();
38 return tr("conflicted with a transaction with %1 confirmations").arg(-nDepth
);
39 else if (GetAdjustedTime() - wtx
.nTimeReceived
> 2 * 60 && wtx
.GetRequestCount() == 0)
40 return tr("%1/offline").arg(nDepth
);
42 return tr("0/unconfirmed, %1").arg((wtx
.InMempool() ? tr("in memory pool") : tr("not in memory pool"))) + (wtx
.isAbandoned() ? ", "+tr("abandoned") : "");
44 return tr("%1/unconfirmed").arg(nDepth
);
46 return tr("%1 confirmations").arg(nDepth
);
50 QString
TransactionDesc::toHTML(CWallet
*wallet
, CWalletTx
&wtx
, TransactionRecord
*rec
, int unit
)
54 LOCK2(cs_main
, wallet
->cs_wallet
);
55 strHTML
.reserve(4000);
56 strHTML
+= "<html><font face='verdana, arial, helvetica, sans-serif'>";
58 int64_t nTime
= wtx
.GetTxTime();
59 CAmount nCredit
= wtx
.GetCredit(ISMINE_ALL
);
60 CAmount nDebit
= wtx
.GetDebit(ISMINE_ALL
);
61 CAmount nNet
= nCredit
- nDebit
;
63 strHTML
+= "<b>" + tr("Status") + ":</b> " + FormatTxStatus(wtx
);
64 int nRequests
= wtx
.GetRequestCount();
68 strHTML
+= tr(", has not been successfully broadcast yet");
69 else if (nRequests
> 0)
70 strHTML
+= tr(", broadcast through %n node(s)", "", nRequests
);
74 strHTML
+= "<b>" + tr("Date") + ":</b> " + (nTime
? GUIUtil::dateTimeStr(nTime
) : "") + "<br>";
81 strHTML
+= "<b>" + tr("Source") + ":</b> " + tr("Generated") + "<br>";
83 else if (wtx
.mapValue
.count("from") && !wtx
.mapValue
["from"].empty())
86 strHTML
+= "<b>" + tr("From") + ":</b> " + GUIUtil::HtmlEscape(wtx
.mapValue
["from"]) + "<br>";
90 // Offline transaction
94 if (CBitcoinAddress(rec
->address
).IsValid())
96 CTxDestination address
= CBitcoinAddress(rec
->address
).Get();
97 if (wallet
->mapAddressBook
.count(address
))
99 strHTML
+= "<b>" + tr("From") + ":</b> " + tr("unknown") + "<br>";
100 strHTML
+= "<b>" + tr("To") + ":</b> ";
101 strHTML
+= GUIUtil::HtmlEscape(rec
->address
);
102 QString addressOwned
= (::IsMine(*wallet
, address
) == ISMINE_SPENDABLE
) ? tr("own address") : tr("watch-only");
103 if (!wallet
->mapAddressBook
[address
].name
.empty())
104 strHTML
+= " (" + addressOwned
+ ", " + tr("label") + ": " + GUIUtil::HtmlEscape(wallet
->mapAddressBook
[address
].name
) + ")";
106 strHTML
+= " (" + addressOwned
+ ")";
116 if (wtx
.mapValue
.count("to") && !wtx
.mapValue
["to"].empty())
118 // Online transaction
119 std::string strAddress
= wtx
.mapValue
["to"];
120 strHTML
+= "<b>" + tr("To") + ":</b> ";
121 CTxDestination dest
= CBitcoinAddress(strAddress
).Get();
122 if (wallet
->mapAddressBook
.count(dest
) && !wallet
->mapAddressBook
[dest
].name
.empty())
123 strHTML
+= GUIUtil::HtmlEscape(wallet
->mapAddressBook
[dest
].name
) + " ";
124 strHTML
+= GUIUtil::HtmlEscape(strAddress
) + "<br>";
130 if (wtx
.IsCoinBase() && nCredit
== 0)
135 CAmount nUnmatured
= 0;
136 BOOST_FOREACH(const CTxOut
& txout
, wtx
.tx
->vout
)
137 nUnmatured
+= wallet
->GetCredit(txout
, ISMINE_ALL
);
138 strHTML
+= "<b>" + tr("Credit") + ":</b> ";
139 if (wtx
.IsInMainChain())
140 strHTML
+= BitcoinUnits::formatHtmlWithUnit(unit
, nUnmatured
)+ " (" + tr("matures in %n more block(s)", "", wtx
.GetBlocksToMaturity()) + ")";
142 strHTML
+= "(" + tr("not accepted") + ")";
150 strHTML
+= "<b>" + tr("Credit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, nNet
) + "<br>";
154 isminetype fAllFromMe
= ISMINE_SPENDABLE
;
155 BOOST_FOREACH(const CTxIn
& txin
, wtx
.tx
->vin
)
157 isminetype mine
= wallet
->IsMine(txin
);
158 if(fAllFromMe
> mine
) fAllFromMe
= mine
;
161 isminetype fAllToMe
= ISMINE_SPENDABLE
;
162 BOOST_FOREACH(const CTxOut
& txout
, wtx
.tx
->vout
)
164 isminetype mine
= wallet
->IsMine(txout
);
165 if(fAllToMe
> mine
) fAllToMe
= mine
;
170 if(fAllFromMe
& ISMINE_WATCH_ONLY
)
171 strHTML
+= "<b>" + tr("From") + ":</b> " + tr("watch-only") + "<br>";
176 BOOST_FOREACH(const CTxOut
& txout
, wtx
.tx
->vout
)
179 isminetype toSelf
= wallet
->IsMine(txout
);
180 if ((toSelf
== ISMINE_SPENDABLE
) && (fAllFromMe
== ISMINE_SPENDABLE
))
183 if (!wtx
.mapValue
.count("to") || wtx
.mapValue
["to"].empty())
185 // Offline transaction
186 CTxDestination address
;
187 if (ExtractDestination(txout
.scriptPubKey
, address
))
189 strHTML
+= "<b>" + tr("To") + ":</b> ";
190 if (wallet
->mapAddressBook
.count(address
) && !wallet
->mapAddressBook
[address
].name
.empty())
191 strHTML
+= GUIUtil::HtmlEscape(wallet
->mapAddressBook
[address
].name
) + " ";
192 strHTML
+= GUIUtil::HtmlEscape(CBitcoinAddress(address
).ToString());
193 if(toSelf
== ISMINE_SPENDABLE
)
194 strHTML
+= " (own address)";
195 else if(toSelf
& ISMINE_WATCH_ONLY
)
196 strHTML
+= " (watch-only)";
201 strHTML
+= "<b>" + tr("Debit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, -txout
.nValue
) + "<br>";
203 strHTML
+= "<b>" + tr("Credit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, txout
.nValue
) + "<br>";
209 CAmount nChange
= wtx
.GetChange();
210 CAmount nValue
= nCredit
- nChange
;
211 strHTML
+= "<b>" + tr("Total debit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, -nValue
) + "<br>";
212 strHTML
+= "<b>" + tr("Total credit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, nValue
) + "<br>";
215 CAmount nTxFee
= nDebit
- wtx
.tx
->GetValueOut();
217 strHTML
+= "<b>" + tr("Transaction fee") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, -nTxFee
) + "<br>";
222 // Mixed debit transaction
224 BOOST_FOREACH(const CTxIn
& txin
, wtx
.tx
->vin
)
225 if (wallet
->IsMine(txin
))
226 strHTML
+= "<b>" + tr("Debit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, -wallet
->GetDebit(txin
, ISMINE_ALL
)) + "<br>";
227 BOOST_FOREACH(const CTxOut
& txout
, wtx
.tx
->vout
)
228 if (wallet
->IsMine(txout
))
229 strHTML
+= "<b>" + tr("Credit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, wallet
->GetCredit(txout
, ISMINE_ALL
)) + "<br>";
233 strHTML
+= "<b>" + tr("Net amount") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, nNet
, true) + "<br>";
238 if (wtx
.mapValue
.count("message") && !wtx
.mapValue
["message"].empty())
239 strHTML
+= "<br><b>" + tr("Message") + ":</b><br>" + GUIUtil::HtmlEscape(wtx
.mapValue
["message"], true) + "<br>";
240 if (wtx
.mapValue
.count("comment") && !wtx
.mapValue
["comment"].empty())
241 strHTML
+= "<br><b>" + tr("Comment") + ":</b><br>" + GUIUtil::HtmlEscape(wtx
.mapValue
["comment"], true) + "<br>";
243 strHTML
+= "<b>" + tr("Transaction ID") + ":</b> " + rec
->getTxID() + "<br>";
244 strHTML
+= "<b>" + tr("Transaction total size") + ":</b> " + QString::number(wtx
.tx
->GetTotalSize()) + " bytes<br>";
245 strHTML
+= "<b>" + tr("Output index") + ":</b> " + QString::number(rec
->getOutputIndex()) + "<br>";
247 // Message from normal bitcoin:URI (bitcoin:123...?message=example)
248 Q_FOREACH (const PAIRTYPE(std::string
, std::string
)& r
, wtx
.vOrderForm
)
249 if (r
.first
== "Message")
250 strHTML
+= "<br><b>" + tr("Message") + ":</b><br>" + GUIUtil::HtmlEscape(r
.second
, true) + "<br>";
253 // PaymentRequest info:
255 Q_FOREACH (const PAIRTYPE(std::string
, std::string
)& r
, wtx
.vOrderForm
)
257 if (r
.first
== "PaymentRequest")
259 PaymentRequestPlus req
;
260 req
.parse(QByteArray::fromRawData(r
.second
.data(), r
.second
.size()));
262 if (req
.getMerchant(PaymentServer::getCertStore(), merchant
))
263 strHTML
+= "<b>" + tr("Merchant") + ":</b> " + GUIUtil::HtmlEscape(merchant
) + "<br>";
267 if (wtx
.IsCoinBase())
269 quint32 numBlocksToMaturity
= COINBASE_MATURITY
+ 1;
270 strHTML
+= "<br>" + tr("Generated coins must mature %1 blocks before they can be spent. When you generated this block, it was broadcast to the network to be added to the block chain. If it fails to get into the chain, its state will change to \"not accepted\" and it won't be spendable. This may occasionally happen if another node generates a block within a few seconds of yours.").arg(QString::number(numBlocksToMaturity
)) + "<br>";
278 strHTML
+= "<hr><br>" + tr("Debug information") + "<br><br>";
279 BOOST_FOREACH(const CTxIn
& txin
, wtx
.tx
->vin
)
280 if(wallet
->IsMine(txin
))
281 strHTML
+= "<b>" + tr("Debit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, -wallet
->GetDebit(txin
, ISMINE_ALL
)) + "<br>";
282 BOOST_FOREACH(const CTxOut
& txout
, wtx
.tx
->vout
)
283 if(wallet
->IsMine(txout
))
284 strHTML
+= "<b>" + tr("Credit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit
, wallet
->GetCredit(txout
, ISMINE_ALL
)) + "<br>";
286 strHTML
+= "<br><b>" + tr("Transaction") + ":</b><br>";
287 strHTML
+= GUIUtil::HtmlEscape(wtx
.tx
->ToString(), true);
289 strHTML
+= "<br><b>" + tr("Inputs") + ":</b>";
292 BOOST_FOREACH(const CTxIn
& txin
, wtx
.tx
->vin
)
294 COutPoint prevout
= txin
.prevout
;
297 if(pcoinsTip
->GetCoins(prevout
.hash
, prev
))
299 if (prevout
.n
< prev
.vout
.size())
302 const CTxOut
&vout
= prev
.vout
[prevout
.n
];
303 CTxDestination address
;
304 if (ExtractDestination(vout
.scriptPubKey
, address
))
306 if (wallet
->mapAddressBook
.count(address
) && !wallet
->mapAddressBook
[address
].name
.empty())
307 strHTML
+= GUIUtil::HtmlEscape(wallet
->mapAddressBook
[address
].name
) + " ";
308 strHTML
+= QString::fromStdString(CBitcoinAddress(address
).ToString());
310 strHTML
= strHTML
+ " " + tr("Amount") + "=" + BitcoinUnits::formatHtmlWithUnit(unit
, vout
.nValue
);
311 strHTML
= strHTML
+ " IsMine=" + (wallet
->IsMine(vout
) & ISMINE_SPENDABLE
? tr("true") : tr("false")) + "</li>";
312 strHTML
= strHTML
+ " IsWatchOnly=" + (wallet
->IsMine(vout
) & ISMINE_WATCH_ONLY
? tr("true") : tr("false")) + "</li>";
320 strHTML
+= "</font></html>";