Merge #10114: [tests] sync_with_ping should assert that ping hasn't timed out
[bitcoinplatinum.git] / src / qt / transactiondesc.cpp
blob7e16cc9dd4130b73ddea0d5738ad647717193535
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"
8 #include "guiutil.h"
9 #include "paymentserver.h"
10 #include "transactionrecord.h"
12 #include "base58.h"
13 #include "consensus/consensus.h"
14 #include "validation.h"
15 #include "script/script.h"
16 #include "timedata.h"
17 #include "util.h"
18 #include "wallet/db.h"
19 #include "wallet/wallet.h"
21 #include <stdint.h>
22 #include <string>
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());
31 else
32 return tr("Open until %1").arg(GUIUtil::dateTimeStr(wtx.tx->nLockTime));
34 else
36 int nDepth = wtx.GetDepthInMainChain();
37 if (nDepth < 0)
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);
41 else if (nDepth == 0)
42 return tr("0/unconfirmed, %1").arg((wtx.InMempool() ? tr("in memory pool") : tr("not in memory pool"))) + (wtx.isAbandoned() ? ", "+tr("abandoned") : "");
43 else if (nDepth < 6)
44 return tr("%1/unconfirmed").arg(nDepth);
45 else
46 return tr("%1 confirmations").arg(nDepth);
50 QString TransactionDesc::toHTML(CWallet *wallet, CWalletTx &wtx, TransactionRecord *rec, int unit)
52 QString strHTML;
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();
65 if (nRequests != -1)
67 if (nRequests == 0)
68 strHTML += tr(", has not been successfully broadcast yet");
69 else if (nRequests > 0)
70 strHTML += tr(", broadcast through %n node(s)", "", nRequests);
72 strHTML += "<br>";
74 strHTML += "<b>" + tr("Date") + ":</b> " + (nTime ? GUIUtil::dateTimeStr(nTime) : "") + "<br>";
77 // From
79 if (wtx.IsCoinBase())
81 strHTML += "<b>" + tr("Source") + ":</b> " + tr("Generated") + "<br>";
83 else if (wtx.mapValue.count("from") && !wtx.mapValue["from"].empty())
85 // Online transaction
86 strHTML += "<b>" + tr("From") + ":</b> " + GUIUtil::HtmlEscape(wtx.mapValue["from"]) + "<br>";
88 else
90 // Offline transaction
91 if (nNet > 0)
93 // Credit
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) + ")";
105 else
106 strHTML += " (" + addressOwned + ")";
107 strHTML += "<br>";
114 // To
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>";
128 // Amount
130 if (wtx.IsCoinBase() && nCredit == 0)
133 // Coinbase
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()) + ")";
141 else
142 strHTML += "(" + tr("not accepted") + ")";
143 strHTML += "<br>";
145 else if (nNet > 0)
148 // Credit
150 strHTML += "<b>" + tr("Credit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit, nNet) + "<br>";
152 else
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;
168 if (fAllFromMe)
170 if(fAllFromMe & ISMINE_WATCH_ONLY)
171 strHTML += "<b>" + tr("From") + ":</b> " + tr("watch-only") + "<br>";
174 // Debit
176 BOOST_FOREACH(const CTxOut& txout, wtx.tx->vout)
178 // Ignore change
179 isminetype toSelf = wallet->IsMine(txout);
180 if ((toSelf == ISMINE_SPENDABLE) && (fAllFromMe == ISMINE_SPENDABLE))
181 continue;
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)";
197 strHTML += "<br>";
201 strHTML += "<b>" + tr("Debit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit, -txout.nValue) + "<br>";
202 if(toSelf)
203 strHTML += "<b>" + tr("Credit") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit, txout.nValue) + "<br>";
206 if (fAllToMe)
208 // Payment to self
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();
216 if (nTxFee > 0)
217 strHTML += "<b>" + tr("Transaction fee") + ":</b> " + BitcoinUnits::formatHtmlWithUnit(unit, -nTxFee) + "<br>";
219 else
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>";
236 // Message
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()));
261 QString merchant;
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>";
274 // Debug view
276 if (fDebug)
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>";
290 strHTML += "<ul>";
292 BOOST_FOREACH(const CTxIn& txin, wtx.tx->vin)
294 COutPoint prevout = txin.prevout;
296 CCoins prev;
297 if(pcoinsTip->GetCoins(prevout.hash, prev))
299 if (prevout.n < prev.vout.size())
301 strHTML += "<li>";
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>";
317 strHTML += "</ul>";
320 strHTML += "</font></html>";
321 return strHTML;